C&C++   发布时间:2022-04-03  发布网站:大佬教程  code.js-code.com
大佬教程收集整理的这篇文章主要介绍了c – 在没有实例的情况下调用无状态lambda(仅限类型)大佬教程大佬觉得挺不错的,现在分享给大家,也给大家做个参考。
我正在尝试从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是无国籍的,所以复制是免费的.

大佬总结

以上是大佬教程为你收集整理的c – 在没有实例的情况下调用无状态lambda(仅限类型)全部内容,希望文章能够帮你解决c – 在没有实例的情况下调用无状态lambda(仅限类型)所遇到的程序开发问题。

如果觉得大佬教程网站内容还不错,欢迎将大佬教程推荐给程序员好友。

本图文内容来源于网友网络收集整理提供,作为学习参考使用,版权属于原作者。
如您有任何意见或建议可联系处理。小编QQ:384754419,请注明来意。