具有/不具有捕获变量的lambda之间的签名差异?

And*_*Dog 6 lambda c++11

我现在正在使用C++ 11,并发现使用lambda作为sqlite回调的以下问题.在lambda中捕获向量变量时,我收到一条错误,指出签名不匹配.不在lambda中使用该变量([]而不是在内部[&ret]使用ret),它可以正常工作.

vector<SomeClass> ret;
char *err = nullptr;
int res = sqlite3_exec(db,
                       "some sql query, doesn't matter",
                       [&ret](void *unused, int argc, char **argv, char **columnName) -> int
                       {
                           ret.push_back(SomeClass());
                           return 0;
                       },
                       nullptr,
                       &err);
Run Code Online (Sandbox Code Playgroud)

这是我得到的错误:

cannot convert 'TestClass::testMethod()::<lambda(void*, int, char**, char**)>' to 'int (*)(void*, int, char**, char**)' for argument '3' to 'int sqlite3_exec(sqlite3*, const char*, int (*)(void*, int, char**, char**), void*, char**)'
Run Code Online (Sandbox Code Playgroud)

GCC版本是Windows上的"gcc(XvidVideo.RU - GCC 4.6.1 i686-pc-mingw32)4.6.1 20110625(预发行版)".

为什么这会有所不同?

Cub*_*bbi 6

只有无捕获的lambda可以转换为指向函数的指针,并且,基于编译器诊断,你sqlite3_exec期望这样的指针,int (*)(void*, int, char**, char**).

引用§5.1.2[expr.prim.lambda]/6

没有lambda-capture的lambda表达式的闭包类型有一个公共的非虚拟非显式const转换函数,用于指向具有与闭包类型的函数调用操作符相同的参数和返回类型的函数.

  • 是的,`sqlite3_exec`允许我将`void*'传递给回调,我只是天真地认为用C++ 11可以更容易地做到这一点. (2认同)