微信公众号搜"智元新知"关注
微信扫一扫可直接关注哦!

c – 将具有默认参数的lambda函数复制到变量

请考虑以下代码
#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);

Live example.

这将创建您的lambda每个重载的单独副本.通过仔细的铸造,甚至可以有不同的羊皮不同的重量.

你可以写一个不这样做的,但它基本上需要重新实现std ::函数一个鸽友的内部状态.

上述的更高级版本将避免使用线性继承,因为对签名数量导致O(n ^ 2)个代码和O(n)递归模板深度.平衡二叉树继承下降到O(n lg n)代码生成和O(lg n)深度.

工业强度版本将存储传入的lambda一次,使用小对象优化,具有使用二进制继承策略来分派函数调用的手动pseudo-vtable,并将调度函数指针存储在所述伪vtable中.在每个类(不是每个实例)的基础上,需要使用O(#签名)* sizeof(函数指针)空间,并且使用与std ::函数相同的每个实例开销.

但是,这对于SO职位来说有点不好吗?

start of it

原文地址:https://www.jb51.cc/c/113452.html

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。

相关推荐