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

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

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

比如,有这个代码

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模板来实现。

template<typename Function>
struct function_traits : public function_traits_help<Function*>
{
};

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

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

void (int,int)//定义一个函数类型
void (*)(int, int) //定义了一个函数指针 template<typename Func>
struct FuncType {
typedef FuncPtr funcPtr;
};
FuncType<void(int,int)>::funcPtr 等同于 void(*)(int,int)

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

首先,定义主模板

template<typename FunctionPtr> struct function_traits_help;

主模板不做任何事情,作为一个核心模板存在,但从不使用,因此无需具体定义。

定义无参数的实现

template<typename R>
struct function_traits_help<R(*)(void)>
{
enum {arty = 0 };
typedef R result_type;
};

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

定义包含一个参数的模板

template<typename R, typename T1>
struct function_traits_help<R(*)(T1)>
{
enum {arty = 1 };
typedef R result_type;
typedef T1 arg1_type;
};

同样的,C++编译器遇到 R (*)(T1)类型的函数指针时,就会匹配到这个定义。

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

template<typename R, typename T1, typename T2>
struct function_traits_help<R(*)(T1, T2)>
{
enum {arty = 2 };
typedef R result_type;
typedef T1 arg1_type;
typedef T2 arg2_type;
}; template<typename R, typename T1, typename T2, typename T3>
struct function_traits_help<R(*)(T1, T2, T3)>
{
enum {arty = 3 };
typedef R result_type;
typedef T1 arg1_type;
typedef T2 arg2_type;
typedef T3 arg3_type;
}; template<typename R, typename T1, typename T2, typename T3, typename T4>
struct function_traits_help<R(*)(T1, T2, T3, T4)>
{
enum {arty = 4 };
typedef R result_type;
typedef T1 arg1_type;
typedef T2 arg2_type;
typedef T3 arg3_type;
typedef T4 arg4_type;
}; ..............

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

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>

template<typename T>
void show_function_info(const char* name)
{
printf("%s: arg count=%d, result_type:%s\n", name, T::arty, typeid(typename T::result_type).name());
show_args<T::arty, T> x;
x();
}

show_args<T::arty, T>也是一个不完全实例化的对象,用于匹配正确数目参数的对象。

show_args的定义如下:

template<int N, typename Func>
struct show_args; //主模板,从不使用 template<typename Func>
struct show_args<0, Func> { //0个参数
void operator()(){ }
}; template<typename Func>
struct show_args<1, Func> {//1个参数
void operator()(){
printf("\targ1 = %s\n", typeid(typename Func::arg1_type).name());
}
}; template<typename Func>
struct show_args<2, Func> {//2个参数
void operator()(){
printf("\targ1 = %s\n", typeid(typename Func::arg1_type).name());
printf("\targ2 = %s\n", typeid(typename Func::arg2_type).name());
}
}; template<typename Func>
struct show_args<3, Func> {
void operator()(){
printf("\targ1 = %s\n", typeid(typename Func::arg1_type).name());
printf("\targ2 = %s\n", typeid(typename Func::arg2_type).name());
printf("\targ3 = %s\n", typeid(typename Func::arg3_type).name());
}
}; template<typename Func>
struct show_args<4, Func> {
void operator()(){
printf("\targ1 = %s\n", typeid(typename Func::arg1_type).name());
printf("\targ2 = %s\n", typeid(typename Func::arg2_type).name());
printf("\targ3 = %s\n", typeid(typename Func::arg3_type).name());
printf("\targ4 = %s\n", typeid(typename Func::arg4_type).name());
}
}; .................................

最后,用宏来简化实现

#define SHOW_FUNC(T) \
show_function_info<T > (#T)

最后的测试代码

int main()
{
SHOW_FUNC(function_traits<int ()>);
SHOW_FUNC(function_traits<void ()>);
SHOW_FUNC(function_traits<void (int)>);
SHOW_FUNC(function_traits<int (int,char*)>);
SHOW_FUNC(function_traits<int (int,char*, double)>); };

该程序的运行结果是

function_traits<int ()>: arg count=0, result_type:i
function_traits<void ()>: arg count=0, result_type:v
function_traits<void (int)>: arg count=1, result_type:v
arg1 = i
function_traits<int (int,char*)>: arg count=2, result_type:i
arg1 = i
arg2 = Pc
function_traits<int (int,char*, double)>: arg count=3, result_type:i
arg1 = i
arg2 = Pc
arg3 = d

C++利用不完全实例化来获得函数模板参数的返回值和参数的更多相关文章

  1. day11 python学习 函数的建立,返回值,参数

    函数的定义主要有如下要点: def:表示函数的关键字 函数名:函数的名称,日后根据函数名调用函数 函数体:函数中进行一系列的逻辑计算,如:发送邮件.计算出 [11,22,38,888,2]中的最大数等 ...

  2. python中函数的定义、返回值以及参数的简要介绍

    一. 1.函数 定义:def关键字开头,空格之后接函数名称和圆括号(),后面紧跟‘:”. 函数是对功能的封装 语法: def 函数名(形参列表): 函数体(代码块,return) 调用: 函数名(实参 ...

  3. python函数进阶(函数参数、返回值、递归函数)

    函数进阶 目标 函数参数和返回值的作用 函数的返回值 进阶 函数的参数 进阶 递归函数 01. 函数参数和返回值的作用 函数根据 有没有参数 以及 有没有返回值,可以 相互组合,一共有 4 种 组合形 ...

  4. Python——变量的引用和函数的参数和返回值的传递方式

    变量的引用 在python中,所有的变量都是指向地址,变量本身不保存数据,而是保存数据在内存中的地址.我们用下面的程序来理解: a = 10 print(id(a)) a = 11 print(id( ...

  5. python中函数的参数和返回值

    目录 函数 目标 01. 函数参数和返回值的作用 1.1 无参数,无返回值 1.2 无参数,有返回值 1.3 有参数,无返回值 1.4 有参数,有返回值 02. 函数的返回值 进阶 示例 -- 温度和 ...

  6. day09-Python运维开发基础(函数收集参数、命名关键字参数与返回值、函数名的特殊使用及 全局/局部变量详解)

    1. 函数收集参数.命名关键字参数与返回值.函数名的特殊使用 # ### 默认形参 和 关键字实参 # 默认形参和 关键字实参 在写法上是一样 # 函数的定义处 """默 ...

  7. 【Go入门教程3】流程(if、goto、for、switch)和函数(多个返回值、变参、传值与传指针、defer、函数作为值/类型、Panic和Recover、main函数和init函数、import)

    这小节我们要介绍Go里面的流程控制以及函数操作. 流程控制 流程控制在编程语言中是最伟大的发明了,因为有了它,你可以通过很简单的流程描述来表达很复杂的逻辑.Go中流程控制分三大类:条件判断,循环控制和 ...

  8. c&c++函数的参数和返回值的传递终结版

    c++函数的参数和返回值的传递方式有三种:值传递.指针传递和引用传递. 在这之前先看几个例子: 一, int a=10; int b=a; b+=10; 此时b是a的一个拷贝,改变b的值,a并不会受到 ...

  9. python笔记六(函数的参数、返回值)

    一 调用函数 在写函数之前,我们先尝试调用现有的函数 >>> abs(-9) 9 除此之外,还有我们之前使用的len()等.可以用于数据类型转换的 int() float() str ...

随机推荐

  1. java环境变量配置四种方法

    原文:java环境变量配置四种方法 Java编程首要工作就是安装JDK(Java Development Kit).一通“NEXT”点完安装后就是最重要的环境变量设置了.也许有人会问为什么要设置环境变 ...

  2. Android-Launcher开发之ShortCut(1)

    下面源代码来自Launcher2.3的样例 1.默认每一个应用的主Activity都会自带 <category android:name="android.intent.categor ...

  3. java ResultSet 结果集处理 createStatement() 里参数的意义(第一弹)

    createStatement(int   resultSetType,int  resultSetConcurrency)参数一:结果集类型可取值:  1.ResultSet.TYPE_FORWOR ...

  4. C#框架

    从零开始编写自己的C#框架(1)——前言   记得十五年前自学编程时,拿着C语言厚厚的书,想要上机都不知道要用什么编译器来执行书中的例子.十二年前在大学自学ASP时,由于身边没有一位同学和朋友学习这种 ...

  5. 在Installshield的安装进度中显示自己设置的信息

    原文:在Installshield的安装进度中显示自己设置的信息 以Installscript msi project为例,在installshield所制作的安装包安装过程中显示安装进度的,就在On ...

  6. 快速构建Windows 8风格应用25-数据绑定

    原文:快速构建Windows 8风格应用25-数据绑定 本篇博文主要介绍如何将UI元素与数据进行绑定.数据绑定的方向.数据更改通知.数据转换.数据绑定支持的绑定方案. 数据绑定是一种简单方式来显示数据 ...

  7. PDFBox之文档创建

    1.创建一个空的PDF 下面的小例子表示如何使用PDFBox来创建一个新的PDF文档. // 创建一个空的文档 PDDocument document = new PDDocument(); // 创 ...

  8. 4GB内存原32位系统(x86)取舍问题,显卡共享内存Win8.1完全不用担心

    情景:集成显卡 配置: 4G显示3.25GB 此时系统自动将用不到的系统完全共享给显卡(768MB而不是256): 看显卡适配器信息,完全共享给了显卡 解说:上图总可用图形内存 = 图2中备用 + 硬 ...

  9. JS中的模块规范(CommonJS,AMD,CMD)

    JS中的模块规范(CommonJS,AMD,CMD) 如果你听过js模块化这个东西,那么你就应该听过或CommonJS或AMD甚至是CMD这些规范咯,我也听过,但之前也真的是听听而已. 现在就看看吧, ...

  10. 【MS SQL】把多个数据库合并为一个新的数据库

    原文:[MS SQL]把多个数据库合并为一个新的数据库 因应工作要求,需要把两个数据库合并成一个库: 一开始使用"导入数据.导出数据和复制数据库"三个工具时,没有达到要的效果. 后 ...