C++ 11将函数作为lambda参数传递

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功能]),那真的很酷......

Yak*_*ont 7

变量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不能).

  • @Marandil Move-into lambdas可以手动编写为帮助程序类型(我承认,它有点冗长).您可以通过将`fn`本地存储在`auto my_fn = std :: forward <Fn>(fn);`,然后捕获`my_fn`来解决衰变问题.请注意,我上面的C++ 14版本(和你的)不必要地将`args ...'复制到lambda中:这是低效的. (2认同)