加入收藏 | 设为首页 | 会员中心 | 我要投稿 李大同 (https://www.lidatong.com.cn/)- 科技、建站、经验、云计算、5G、大数据,站长网!
当前位置: 首页 > 百科 > 正文

c – 在没有实例的情况下调用无状态lambda(仅限类型)

发布时间:2020-12-16 10:11:29 所属栏目:百科 来源:网络整理
导读:我正在尝试从C库编写一个 “register callback” type of interface的包装器.这个问题非常复杂,因为库允许您通过接受参数定义列表来注册“可变参数”函数.然后在回调时,函数应该从类型擦除的参数列表中提取其参数.好老C … 我正在尝试创建的接口是接受任何函
我正在尝试从C库编写一个 “register callback” type of interface的包装器.这个问题非常复杂,因为库允许您通过接受参数定义列表来注册“可变参数”函数.然后在回调时,函数应该从类型擦除的参数列表中提取其参数.好老C …

我正在尝试创建的接口是接受任何函数,甚至是lambda,并自动生成所有机制以正确注册此函数,同时注册参数并在回调时提取它们.因此用户只需键入:

register_callback("Callback Name",[](int i){return i+0.5;});

现在这在理论上是可行的,但是在某一点上,我需要能够在没有任何实例的情况下调用lambda,但只能访问它的类型.我不明白为什么,但是无状态(非捕获)lambda的默认构造函数被删除,所以我不能简单地从它的类型默认构造.我打算做的事情看起来很脏,但应该可以解决这个问题:

template <class Func,class RET,class ...ARGS>
struct from_stateless {
    static RET to_function(ARGS...args) {
        RET (*f)(ARGS...) = *(Func *)(0);
        return f(args...);
    }
};

所以from_stateless< LAMBDA_TYPE,RET,ARGS ...> :: to_function实际上是一个函数指针,我可以在没有参数的情况下调用,更重要的是我可以作为模板参数传递的函数指针.

在正常情况下,行RET(* f)(ARGS …)= *(Func *)(0);会是自杀,但这个用例确实应该是安全的,不是吗?毕竟,转换后获得的函数指针不能在任何可能依赖于lambda实例的Universe中.

所以,问题是,只要我确保该类型确实是无状态lambda,这样做是否安全?或者我错过了什么?注意RET(* f)(ARGS …)= *(Func *)(0);如果闭包意外滑过,将触发编译器错误.

澄清:我不能将lambda衰减为函数指针并注册它,因为lambda的签名与“Register”方法不兼容. register方法需要一个带签名的函数:void(*)(int number_of_args,TypeErasedValue * arguments,TypeErasedValue * result)所以你看,我需要(并且已经做),通过模板元编程,生成一个模板化的自由函数lambda用作预期签名和实际签名之间的适配器.

解决方法

想要定义的行为?

加:

static Func state(Func* op=0){
  static Func f=*op;
  return f;
}

到您的模板类.第一次调用它时,将ptr传递给lambda.第二次,免费提取价值.

静态Func f是在第一次调用函数时构造的:只要指针在那个时间非空,我们就是好的.每次调用它时,指针都可以为null,并且不使用它,因为静态局部变量只构造一次.

使用指向lambda的指针将其命名为“register”.在其他地方(回调),调用它来获取副本. Lambda是无国籍的,所以复制是免费的.

(编辑:李大同)

【声明】本站内容均来自网络,其相关言论仅代表作者个人观点,不代表本站立场。若无意侵犯到您的权利,请及时与联系站长删除相关内容!

    推荐文章
      热点阅读