C++利用不完全实例化来获得函数模板参数的返回值和参数

 

C++利用不完全实例化来获得函数模板参数的返回值和参数

分类: 开发模式探索 linux 嵌入式 boost   107人阅读  评论(0)  收藏  举报

有一些模板会以函数为模板参数,有时候这些模板要获得函数的返回值和参数。如在boost中的signal和slot机制,就存在这样情况。

那么,我们如何得到这些信息呢?

我们使用C++不完全实例化来实现。


比如,有这个代码

[cpp]  view plain copy
  1. typedef function_traits<void (int,const char*)> Signal;  
能够得到Signal::result_type == void, Signal::arg1_type == int, Signal::arg2_type == const char*?


要想获得这种效果,必须通过Function的指针来实现,我们借助一个function_traits_help模板来实现。

[cpp]  view plain copy
  1. template<typename Function>  
  2. struct function_traits : public function_traits_help<Function*>  
  3. {  
  4. };  

function_traits_help类接受Function的指针为参数。

函数类型和函数指针是不一样的,如

[cpp]  view plain copy
  1. void (int,int)//定义一个函数类型  
  2. void (*)(intint//定义了一个函数指针  
  3.   
  4. template<typename Func>  
  5. struct FuncType {  
  6.    typedef FuncPtr funcPtr;  
  7. };  
  8. FuncType<void(int,int)>::funcPtr 等同于 void(*)(int,int)  


function_traits_help就是利用不完全实例化来实现

首先,定义主模板

[cpp]  view plain copy
  1. template<typename FunctionPtr> struct function_traits_help;  
主模板不做任何事情,作为一个核心模板存在,但从不使用,因此无需具体定义。


定义无参数的实现

[cpp]  view plain copy
  1. template<typename R>  
  2. struct function_traits_help<R(*)(void)>  
  3. {  
  4.     enum {arty = 0 };  
  5.     typedef R result_type;  
  6. };  

function_traits_help<R(*)(void)>就是function_traits_help<FunctionPtr>的一种具体实例化,C++编译器当遇到 R (*)(void)这样类型的函数指针类型的时候,就会匹配到这个定义上。


定义包含一个参数的模板

[cpp]  view plain copy
  1. template<typename R, typename T1>  
  2. struct function_traits_help<R(*)(T1)>  
  3. {  
  4.     enum {arty = 1 };  
  5.     typedef R result_type;  
  6.     typedef T1 arg1_type;  
  7. };  
同样的,C++编译器遇到 R (*)(T1)类型的函数指针时,就会匹配到这个定义。


同理,我们可以定义2,3,4,...

[cpp]  view plain copy
  1. template<typename R, typename T1, typename T2>  
  2. struct function_traits_help<R(*)(T1, T2)>  
  3. {  
  4.     enum {arty = 2 };  
  5.     typedef R result_type;  
  6.     typedef T1 arg1_type;  
  7.     typedef T2 arg2_type;  
  8. };  
  9.   
  10. template<typename R, typename T1, typename T2, typename T3>  
  11. struct function_traits_help<R(*)(T1, T2, T3)>  
  12. {  
  13.     enum {arty = 3 };  
  14.     typedef R result_type;  
  15.     typedef T1 arg1_type;  
  16.     typedef T2 arg2_type;  
  17.     typedef T3 arg3_type;  
  18. };  
  19.   
  20. template<typename R, typename T1, typename T2, typename T3, typename T4>  
  21. struct function_traits_help<R(*)(T1, T2, T3, T4)>  
  22. {  
  23.     enum {arty = 4 };  
  24.     typedef R result_type;  
  25.     typedef T1 arg1_type;  
  26.     typedef T2 arg2_type;  
  27.     typedef T3 arg3_type;  
  28.     typedef T4 arg4_type;  
  29. };  
  30.   
  31. ..............  

这样,我们就可以得到正确的信息:

[html]  view plain copy
  1. typedef function_traits<int (int,char*)> Function;  
于是:

Function::arty  : 2;

Function::result_type : int

Function::arg1_type : int

Function::arg2_type : char* 



我们还可以用typeid(x).name()来看真正的效果。

PS: 要使用typeid(x).name()要首先  #include <typeinfo>


[cpp]  view plain copy
  1. template<typename T>  
  2. void show_function_info(const char* name)  
  3. {  
  4.     printf("%s: arg count=%d, result_type:%s\n", name, T::arty, typeid(typename T::result_type).name());  
  5.     show_args<T::arty, T> x;  
  6.     x();  
  7. }  
show_args<T::arty, T>也是一个不完全实例化的对象,用于匹配正确数目参数的对象。

show_args的定义如下:

[cpp]  view plain copy
  1. template<int N, typename Func>  
  2. struct show_args; //主模板,从不使用  
  3.   
  4. template<typename Func>  
  5. struct show_args<0, Func> { //0个参数  
  6.     void operator()(){ }  
  7. };  
  8.   
  9. template<typename Func>  
  10. struct show_args<1, Func> {//1个参数  
  11.     void operator()(){   
  12.         printf("\targ1 = %s\n"typeid(typename Func::arg1_type).name());  
  13.     }  
  14. };  
  15.   
  16. template<typename Func>  
  17. struct show_args<2, Func> {//2个参数  
  18.     void operator()(){  
  19.         printf("\targ1 = %s\n"typeid(typename Func::arg1_type).name());  
  20.         printf("\targ2 = %s\n"typeid(typename Func::arg2_type).name());  
  21.     }  
  22. };  
  23.   
  24. template<typename Func>  
  25. struct show_args<3, Func> {  
  26.     void operator()(){  
  27.         printf("\targ1 = %s\n"typeid(typename Func::arg1_type).name());  
  28.         printf("\targ2 = %s\n"typeid(typename Func::arg2_type).name());  
  29.         printf("\targ3 = %s\n"typeid(typename Func::arg3_type).name());  
  30.     }  
  31. };  
  32.   
  33. template<typename Func>  
  34. struct show_args<4, Func> {  
  35.     void operator()(){  
  36.         printf("\targ1 = %s\n"typeid(typename Func::arg1_type).name());  
  37.         printf("\targ2 = %s\n"typeid(typename Func::arg2_type).name());  
  38.         printf("\targ3 = %s\n"typeid(typename Func::arg3_type).name());  
  39.         printf("\targ4 = %s\n"typeid(typename Func::arg4_type).name());  
  40.     }  
  41. };  
  42.   
  43. .................................  

最后,用宏来简化实现

[cpp]  view plain copy
  1. #define SHOW_FUNC(T) \  
  2.     show_function_info<T  > (#T)  

最后的测试代码

[cpp]  view plain copy
  1. int main()  
  2. {  
  3.     SHOW_FUNC(function_traits<int ()>);  
  4.     SHOW_FUNC(function_traits<void ()>);  
  5.     SHOW_FUNC(function_traits<void (int)>);  
  6.     SHOW_FUNC(function_traits<int (int,char*)>);  
  7.     SHOW_FUNC(function_traits<int (int,char*, double)>);  
  8.   
  9. };  

该程序的运行结果是

[plain]  view plain copy
  1. function_traits<int ()>: arg count=0, result_type:i  
  2. function_traits<void ()>: arg count=0, result_type:v  
  3. function_traits<void (int)>: arg count=1, result_type:v  
  4.     arg1 = i  
  5. function_traits<int (int,char*)>: arg count=2, result_type:i  
  6.     arg1 = i  
  7.     arg2 = Pc  
  8. function_traits<int (int,char*, double)>: arg count=3, result_type:i  
  9.     arg1 = i  
  10.     arg2 = Pc  
  11.     arg3 = d  
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
提供的源码资源涵盖了Java应用等多个领域,每个领域都包含了丰富的实例和项目。这些源码都是基于各自平台的最新技术和标准编写,确保了在对应环境下能够无缝运行。同时,源码中配备了详细的注释和文档,帮助用户快速理解代码结构和实现逻辑。 适用人群: 适合毕业设计、课程设计作业。这些源码资源特别适合大学生群体。无论你是计算机相关专业的学生,还是对其他领域编程感兴趣的学生,这些资源都能为你提供宝贵的学习和实践机会。通过学习和运行这些源码,你可以掌握各平台开发的基础知识,提升编程能力和项目实战经验。 使用场景及目标: 在学习阶段,你可以利用这些源码资源进行课程实践、课外项目或毕业设计。通过分析和运行源码,你将深入了解各平台开发的技术细节和最佳实践,逐步培养起自己的项目开发和问题解决能力。此外,在求职或创业过程中,具备跨平台开发能力的大学生将更具竞争力。 其他说明: 为了确保源码资源的可运行性和易用性,特别注意了以下几点:首先,每份源码都提供了详细的运行环境和依赖说明,确保用户能够轻松搭建起开发环境;其次,源码中的注释和文档都非常完善,方便用户快速上手和理解代码;最后,我会定期更新这些源码资源,以适应各平台技术的最新发展和市场需求。 所有源码均经过严格测试,可以直接运行,可以放心下载使用。有任何使用问题欢迎随时与博主沟通,第一时间进行解答!

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值