作者热门文章
- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我有一个要用于在EventQueue上执行回调的类:
template <class, class>
class Handler;
template <class Fun, class... Args>
class Handler<void(Args...)>
{
public:
Handler(events::EventQueue &eq, Fun fun) : _eq(eq), _fun(fun)
{}
void operator()(Args... args)
{
_eq.call(this, &Handler::callInternal, args...);
}
private:
events::EventQueue &_eq;
Fun _fun;
void callInternal(Args... args)
{
_fun(args...);
}
};
其中EventQueue::call(...)带有对象指针,方法指针以及用于调用方法的参数。
makeHandler([](int result) { ... });
我不在乎是否必须使用诸如
makeHandler
之类的辅助方法来构造Handler。
template <class Fun, class... Args>
Handler<Fun, void(Args...)> makeHandlerInternal(events::EventQueue &eq, Fun fun, void (Fun::*)(Args...) const)
{
return Handler<Fun, void(Args...)>(eq, fun);
}
template <class Fun, class... Args>
Handler<Fun, void(Args...)> makeHandler(events::EventQueue &eq, Fun fun)
{
return Handler<Fun, void(Args...)>(eq, fun, &Fun::operator());
}
因为Args ...在我将其定义为Fun::operator()接受的参数之前,似乎被推导为空列表。
最佳答案
您的语法几乎是正确的。我看不出将Args...
转换为void(Args...)
并提出以下解决方案的充分理由:
// This stub is here for the sake of code completeness only
class EventQueue {
public:
template<class T, class Fun, class... Args>
void call(T* obj, Fun fun, Args... args) const {
std::invoke(fun, obj, args...);
}
};
template<class Fun, class... Args>
class Handler {
public:
Handler(EventQueue& eq, Fun fun) : eq_(eq), fun_(fun) {}
void operator()(Args... args) {
eq_.call(this, &Handler::callInternal, args...);
}
private:
EventQueue& eq_;
Fun fun_;
void callInternal(Args... args) {
fun_(args...);
}
};
template<class Fun, class... Args>
auto makeHandlerImpl(EventQueue& eq, Fun fun, void (Fun::*)(Args...) const) {
return Handler<Fun, Args...>(eq, fun);
}
template<class Fun>
auto makeHandler(EventQueue& eq, Fun fun) {
return makeHandlerImpl(eq, fun, &Fun::operator());
}
EventQueue eq;
auto handler = makeHandler(eq, [](int result) { std::cout << result; });
handler(2020); // prints 2020
Args...
类型中删除
Handler
,但将
operator()
用作模板,得到如下所示:
template<class Fun>
class Handler {
public:
Handler(EventQueue& eq, Fun fun) : eq_(eq), fun_(fun) {}
template<class... Args>
void operator()(Args... args) {
// static_assert(std::is_invocable_v<Fun, Args...>);
eq_.call(this, &Handler::callInternal<Args...>, args...);
}
private:
EventQueue& eq_;
Fun fun_;
template<class... Args>
void callInternal(Args... args) {
fun_(args...);
}
};
现在,使用C++ 17
CTAD,您可以简单地编写:
EventQueue eq;
Handler handler(eq, [](int result) { std::cout << result; });
handler(2020); // prints 2020
没有任何特殊的帮助功能。
关于c++ - 推导Lambda仿函数的参数包参数,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/63151112/
我是一名优秀的程序员,十分优秀!