【问题标题】:Lambda captures don't seem to work [duplicate]Lambda 捕获似乎不起作用[重复]
【发布时间】:2018-02-06 22:47:53
【问题描述】:

这是一个愚蠢的小程序(在我的 real 应用程序中,'iteration' 试图扫描一个列表,对每个元素执行一些操作)。

#include <stdio.h>

void iteration(
        int& foo, int& bar,
        int (*callback)(int a,int b)) {
    while (foo<100) {
        int tmp = bar;
        bar = callback(foo, bar);
        foo = tmp;
        printf("%d\n", foo);
    }
}

void fibonacci(int foo, int bar) {
    int count = 0;
    iteration(
        foo, bar,
        [&](int a, int b) {
//          count += 1;
            return a+b;
        } );
    printf("count %d\n", count);
}

int main() {
fibonacci(1, 1);
}
// g++ -g -std=c++11 -o fib fib.cpp

当编译(见最后一行的注释)并运行时,它会打印一个简短的斐波那契数列,当然,后面是“count 0”。

现在删除“count += 1;”之前的“//”然后再试一次。编译失败

$ g++ -g -std=c++11 -o fib fib.cpp
fib.cpp: In function ‘void fibonacci(int, int)’:
fib.cpp:21:5: error: cannot convert ‘fibonacci(int, int)::__lambda0’
to ‘int (*)(int, int)’ for argument ‘3’ to 
‘void iteration(int&, int&, int (*)(int, int))’
   } );
     ^

我做错了什么?

我的编译器版本见

$ g++ -v
Using built-in specs.
COLLECT_GCC=g++
COLLECT_LTO_WRAPPER=/usr/lib/gcc/i686-linux-gnu/4.8/lto-wrapper
Target: i686-linux-gnu
Configured with: ../src/configure -v 
--with-pkgversion='Ubuntu 4.8.4-2ubuntu1~14.04.3' 
--with-bugurl=file:///usr/share/doc/gcc-4.8/README.Bugs 
--enable-languages=c,c++,java,go,d,fortran,objc,obj-c++ --prefix=/usr
--program-suffix=-4.8 --enable-shared --enable-linker-build-id 
--libexecdir=/usr/lib --without-included-gettext 
--enable-threads=posix --with-gxx-include-dir=/usr/include/c++/4.8 
--libdir=/usr/lib --enable-nls --with-sysroot=/ 
--enable-clocale=gnu --enable-libstdcxx-debug 
--enable-libstdcxx-time=yes --enable-gnu-unique-object 
--disable-libmudflap --enable-plugin --with-system-zlib 
--disable-browser-plugin --enable-java-awt=gtk --enable-gtk-cairo 
--with-java-home=/usr/lib/jvm/java-1.5.0-gcj-4.8-i386/jre 
--enable-java-home 
--with-jvm-root-dir=/usr/lib/jvm/java-1.5.0-gcj-4.8-i386 
--with-jvm-jar-dir=/usr/lib/jvm-exports/java-1.5.0-gcj-4.8-i386 
--with-arch-directory=i386 
--with-ecj-jar=/usr/share/java/eclipse-ecj.jar 
--enable-objc-gc --enable-targets=all --enable-multiarch 
--disable-werror --with-arch-32=i686 
--with-multilib-list=m32,m64,mx32 --with-tune=generic 
--enable-checking=release 
--build=i686-linux-gnu --host=i686-linux-gnu 
--target=i686-linux-gnu
Thread model: posix
gcc version 4.8.4 (Ubuntu 4.8.4-2ubuntu1~14.04.3) 

【问题讨论】:

  • 不可能使用任何类型的捕获同时保持与“经典”C 风格函数指针的兼容性。

标签: c++ lambda capture


【解决方案1】:

一般来说,只有无捕获的 lambda 可以转换为函数指针。

尝试改用std::function

void iteration(int& foo, int& bar, std::function<int(int, int)> callback) {
    while (foo<100) {
        int tmp = bar;
        bar = callback(foo, bar);
        foo = tmp;
        printf("%d\n", foo);
    }
}

附:我不认为你的第一个程序works well

【讨论】:

  • 好的,效果很好(虽然我还没有将它移植到我的真实应用程序中)。谢谢你。不知道为什么你认为我的第一个程序不起作用。使用 C++11 对我来说效果很好。我注意到您的链接使用的是 c++17。
猜你喜欢
  • 1970-01-01
  • 2011-04-25
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2013-02-06
  • 2013-06-19
  • 2020-06-23
  • 1970-01-01
相关资源
最近更新 更多