省略号和可变参数模板

1.基本概念

  省略号在C/C++中有很多用途,包括函数的变量参数列表。C运行库的printf()就是常见示例。

可变参数模板提供了类型安全和灵活性,可应用于类模板和函数模板。

2.语法示例

template<typename ... args>

可变参数模板可以有0个或多个参数,如果要求至少一个参数:

template<typename one,typename... rest> class classname;

可变参数模板函数示例:

template<typename... Args> returnType fun(Args... args);             //参数按值传递
templata<typename... Args> returnType fun(Args&... args);            //参数引用传递
templata<typename... Args> returnType fun(Args*... args);            //参数为指针类型
templata<typename... Args> returnType fun(const Args&... args);      //常量引用
templata<typename one,typename... Args> returnType fun(const one& arg1,const Args&... args); //需要至少一个参数

 程序示例:

template<typename... Arguments>  
void tfunc(const Arguments&... args)  
{  
    const unsigned numargs = sizeof...(Arguments);  
  
    X xobj[numargs]; // array of some previously defined type X  
  
    helper_func(xobj, args...);  
}  

 

3. C中的省略号

  在C中使用省略号显得比较麻烦,示例如下:

#include <stdio.h>
#include <stdarg.h>

void test(int a,...); 

int main(){
    
    test(1, 2, 3, 4);
    return 0;
}
void test(int a,...)
{
    va_list args;
    int nArg = a; //a作为固定参数,用来确定变参的起始地址
    va_start(args, a);
    printf("%d\n", va_arg(args,int));
    printf("%d\n", va_arg(args, int));
    printf("%d\n", va_arg(args, int));
    va_end(args);
    
}

 

  其中test函数是个简单的示例,在C中使用省略号必须要有一个指定参数(第一位参数),不能只有省略号。由于函数的参数入栈是从右往左入的,栈的生长是从高地址向低地址生长,所以在参数最左(第一位参数)指定参数就方便确定可变参数的地址了 ,不然没法取得可变参数。

  接下来va_list是指向参数的指针,在vadefs.h中定义如下:

typedef char *  va_list;

  然后看一下va_start的定义:

#define _crt_va_start(ap,v)  ( ap = (va_list)_ADDRESSOF(v) + _INTSIZEOF(v) )

  从该定义中可以看出来args指向了第二个参数2

  然后下一句是执行了va_arg(args,int),然后打印

  看一下它的定义

#define _crt_va_arg(ap,t)    ( *(t *)((ap += _INTSIZEOF(t)) - _INTSIZEOF(t)) )

  这句话是将args指向了下一个参数,但是输出的是下一个参数的上一个:( -_INTSIZEOF(t)),这样可以从第二个参数开始,因为va_start已经将args指向了第二个参数,如果不( -_INTSIZEOF(t))的话会跳过第二个,因为args是持续累加的,所以下面的va_arg(args,int)会连续指向第三个参数、第四个......

  然后最后的va_end(args)是释放args的指向

#define _crt_va_end(ap)      ( ap = (va_list)0 )

 

  但是使用va_arg的话会一直指向下一个,由于不知道参数的个数,早晚会内存溢出,这个问题还不太懂,留坑。。。

 

posted @ 2017-02-28 13:48  whlook  阅读(1307)  评论(0编辑  收藏  举报