Mar*_*dil 9 c++ gcc clang c++11 clang++
最近,我遇到了一个奇怪的问题,即将作为参数传递的函数传递给lambda表达式.使用clang 3.5+编译的代码很好,但是使用g ++ 5.3失败了,我想知道问题是在c ++标准中,clang中的非标准扩展,还是GCC中的无效语法解释.
示例代码很简单:
template<typename Fn, typename... Args, typename T = typename std::result_of<Fn(Args...)>::type>
std::future<T>
async(Fn &&fn, Args &&... args)
{
std::shared_ptr<std::promise<T>> promise = std::make_shared<std::promise<T>>();
auto lambda = [promise, fn, args...](void)
{ promise->set_value(fn(std::move(args)...)); };
send_message(std::make_shared<post_call>(lambda));
return promise->get_future();
};
Run Code Online (Sandbox Code Playgroud)
海湾合作委员会报告如下:
error: variable ‘fn’ has function type
{ promise->set_value(fn(std::move(args)...)); };
(...)
error: field ‘async(Fn&&, Args&& ...) [with Fn = int (&)(int, int); Args = {int&, int&}; T = int]::<lambda()>::<fn capture>’ invalidly declared function type
auto lambda = [promise, fn, args...](void)
Run Code Online (Sandbox Code Playgroud)
幸运的是,我找到了一个简单的解决方法,添加了一个std :: function对象,封装了function参数:
template<typename Fn, typename... Args, typename T = typename std::result_of<Fn(Args...)>::type>
std::future<T>
async(Fn &&fn, Args &&... args)
{
std::shared_ptr<std::promise<T>> promise = std::make_shared<std::promise<T>>();
std::function<T(typename std::remove_reference<Args>::type...)> floc = fn;
auto lambda = [promise, floc, args...](void)
{ promise->set_value(floc(std::move(args)...)); };
send_message(std::make_shared<post_call>(lambda));
return promise->get_future();
};
Run Code Online (Sandbox Code Playgroud)
虽然我没有完全理解第一段代码中的错误,但是使用clang成功编译并运行时没有错误.
编辑
我刚刚注意到,如果其中一个参数被认为是一个参考,我的解决方案就会发生灾难性的失败.因此,如果您有任何其他可能适用于C++ 11的建议(即没有专门的lambda捕获[c ++ 14功能]),那真的很酷......
变量fn具有函数类型.特别是它有类型Fn = int (&)(int, int).
类型Fn(不是引用)的类型是类型int(int,int).
无法存储此值.
我模糊地惊讶它不会为你自动腐烂.在C++ 14中,您可以:
auto lambda = [promise, fn=fn, args...](void)
{ promise->set_value(fn(std::move(args)...)); };
Run Code Online (Sandbox Code Playgroud)
哪个应该在内部衰减fn(外部)的类型fn.如果这不起作用:
auto lambda = [promise, fn=std::decay_t<Fn>(fn), args...](void)
{ promise->set_value(fn(std::move(args)...)); };
Run Code Online (Sandbox Code Playgroud)
它明确地腐朽了它.(Decay是一种适合存储的类型的操作).
其次,你应该添加mutable:
auto lambda = [promise, fn=std::decay_t<Fn>(fn), args...](void)mutable
{ promise->set_value(fn(std::move(args)...)); };
Run Code Online (Sandbox Code Playgroud)
或者std::move不会做太多.(移动const值并没有太大作用).
第三,你可以转移承诺,而不是创建一个不必要的共享ptr:
template<typename Fn, typename... Args, typename T = typename std::result_of<Fn(Args...)>::type>
std::future<T>
async(Fn &&fn, Args &&... args)
{
std::promise<T> promise;
std::future<T> ret = promise.get_future();
auto lambda =
[promise=std::move(promise), fn=std::decay_t<Fn>(fn), args...]
() mutable {
promise.set_value(fn(std::move(args)...));
};
send_message(std::make_shared<post_call>(lambda));
return ret;
};
Run Code Online (Sandbox Code Playgroud)
这假设你的post_call班级可以处理只移动的lambda(如果它std::function不能).
| 归档时间: |
|
| 查看次数: |
1742 次 |
| 最近记录: |