【发布时间】:2015-12-05 04:30:45
【问题描述】:
我正在研究一个可能被称为“重载 lambda”的 C++11 习语:
- http://cpptruths.blogspot.com/2014/05/fun-with-lambdas-c14-style-part-2.html
- http://martinecker.com/martincodes/lambda-expression-overloading/
使用可变参数模板重载 n 函数对我来说似乎很有吸引力,但事实证明它不适用于变量捕获:[&][=][y][&y] 中的任何一个(和[this] 等如果在成员函数中)导致编译失败:error: no match for call to '(overload<main(int, char**)::<lambda(int)>, main(int, char**)::<lambda(char*)> >) (char*&)'(使用我本地的 GCC 4.9.1 和 ideone.com GCC 5.1)
另一方面,固定的二元情况并没有遇到这个问题。 (尝试在 ideone.com 上将第一个 #if 0 更改为 #if 1)
对这里发生的事情有什么想法吗?这是编译器错误,还是我偏离了 C++11/14 规范?
#include <iostream>
using namespace std;
#if 0
template <class F1, class F2>
struct overload : F1, F2 {
overload(F1 f1, F2 f2) : F1(f1), F2(f2) { }
using F1::operator();
using F2::operator();
};
template <class F1, class F2>
auto make_overload(F1 f1, F2 f2) {
return overload<F1, F2>(f1, f2);
}
#else
template <class... Fs>
struct overload;
template <class F0, class... Frest>
struct overload<F0, Frest...> : F0, overload<Frest...> {
overload(F0 f0, Frest... rest) : F0(f0), overload<Frest...>(rest...) {}
using F0::operator();
};
template <>
struct overload<> {
overload() {}
};
template <class... Fs>
auto make_overload(Fs... fs) {
return overload<Fs...>(fs...);
}
#endif
#if 0
#define CAP
#define PRINTY()
#else
#define CAP y
#define PRINTY() cout << "int y==" << y << endl
#endif
int main(int argc, char *argv[]) {
int y = 123;
auto f = make_overload(
[CAP] (int x) { cout << "int x==" << x << endl; PRINTY(); },
[CAP] (char *cp) { cout << "char *cp==" << cp << endl; PRINTY(); });
f(argc);
f(argv[0]);
}
【问题讨论】:
-
你应该在第二个实现中拥有
using overload<Frest...>::operator();,demo -
@PiotrSkotnicki 很有趣,但添加
using overload<Frest...>::operator();并没有改善这种情况。 -
啊,你是对的,我没有仔细阅读来自在线编译器的错误信息......缺少的
using overload<Frest...>::operator();是的关键。我还必须将基本情况定义为一元:template <class F0> struct overload<F0> : F0 {overload(F0 f0) : F0(f0) {} using F0::operator();};我会接受您的 cmets 作为解决方案! -
“一个非捕获的 lambda 定义了一个转换运算符”我不知道 en.cppreference.com/w/cpp/language/…
标签: c++ c++11 lambda variadic-templates template-meta-programming