请考虑以下代码:
#include <iostream> #include <functional> using namespace std; int main() { auto f = [](int a = 3) {cout << a << endl; }; f(2); // OK f(); // OK auto g = f; g(2); // OK g(); // OK function<void(int)> h = f; h(2); // OK h(); // Error! How to make this work? return 0; }
我如何声明h的行为与f和g相同?
解决方法
std :: function有一个固定签名.这是一个设计选择,而不是一个艰巨的要求.编写支持多个签名的伪std ::函数并不困难:
template<class...Sigs> struct functions; template<> struct functions<> { functions()=default; functions(functions const&)=default; functions(functions&&)=default; functions& operator=(functions const&)=default; functions& operator=(functions&&)=default; private: struct never_t {private:never_t(){};}; public: void operator()(never_t)const =delete; template<class F,std::enable_if_t<!std::is_same<std::decay_t<F>,functions>{},int>* =nullptr > functions(F&&) {} }; template<class S0,class...Sigs> struct functions<S0,Sigs...>: std::function<S0>,functions<Sigs...> { functions()=default; functions(functions const&)=default; functions(functions&&)=default; functions& operator=(functions const&)=default; functions& operator=(functions&&)=default; using std::function<S0>::operator(); using functions<Sigs...>::operator(); template<class F,int>* =nullptr > functions(F&& f): std::function<S0>(f),functions<Sigs...>(std::forward<F>(f)) {} };
使用:
auto f = [](int a = 3) {std::cout << a << std::endl; }; functions<void(int),void()> fs = f; fs(); fs(3);
这将创建您的lambda每个重载的单独副本.通过仔细的铸造,甚至可以有不同的羊皮不同的重量.
你可以写一个不这样做的,但它基本上需要重新实现std ::函数与一个鸽友的内部状态.
上述的更高级版本将避免使用线性继承,因为对签名数量导致O(n ^ 2)个代码和O(n)递归模板深度.平衡二叉树继承下降到O(n lg n)代码生成和O(lg n)深度.
工业强度版本将存储传入的lambda一次,使用小对象优化,具有使用二进制继承策略来分派函数调用的手动pseudo-vtable,并将调度函数指针存储在所述伪vtable中.在每个类(不是每个实例)的基础上,需要使用O(#签名)* sizeof(函数指针)空间,并且使用与std ::函数相同的每个实例开销.
但是,这对于SO职位来说有点不好吗?