C/C++可变参数列表

可变参数列表

  • 可变参数宏--__VA_ARGS__
  • C风格不定参使用
    • 补充知识:函数调用时参数的压栈顺序及内存使用
    • 使用不定参模拟实现printf
  • C++风格不定参数的使用

可变参数宏–VA_ARGS

#include 

//...表示不定参,__VA_ARGS__使用不定参
// __FILE__ 文件名
//__LINE__ 行号
//__VA_ARGS__ 用于在宏替换部分中,表示可变参数列表;
//当可变参数的个数为0时,##起到把前面多余的","去掉的作用,否则会编译出错
#define LOG(fmt,...) printf("[%s:%d]" fmt,__FILE__,__LINE__,##__VA_ARGS__);

int main()
{
    LOG("可变参数列表");
    //最终展开的代码相当于printf("[%s:%d]" "可变参数列表",__FILE__,__LINE__)
    //如果不加##,会报错
    //printf("%d""nihao",6);可以编译通过
    return 0;
}

在这里我之前一直有一个误区:
1.我认为printf("%d""nihao",6);是不合法的,因为格式化参数后面没有加,,直到今天认识了可变参数宏,才认识到
2."%d""nihao"会自动连接

C风格不定参使用

使用之前需要自己定义一个va_list 变量
va_start()函数 语法

void va_start(va_list ap,param);
参数ap为参数自身;
参数param为第一个参数。

va_arg()函数用于调用可变参数列表

type va_arg(va_list ap,type);
参数ap为可变参数自身;

参数type为要获取的参数的指定类型,返回这个指定类型的值,并把ap的位置指向变参表的下一个变量位置

va_end()函数用于停止使用可变参数。

void va_end(va_list ap);
参数ap为参数自身。

简单使用

void print(int cnt,...)
{
    va_list ap;
    va_start(ap,cnt);//获取cnt参数之后的第一个参数的地址即第一个不定参地址
    for(int i=0;i<cnt;++i)
    {
        int num=va_arg(ap,int);//获取不定参
        printf("param[%d]:%d\n",i,num);
    }
    va_end(ap);//使用完将ap指针置空

}

补充知识:函数调用时参数的压栈顺序及内存使用

printf(const char* format,…)为例

  1. printf函数是一个不定参函数。
  2. 编译器通过format的%占位符的个数来获取参数的个数。
  3. 假设函数压栈顺序是从左至右,format先入栈,各个参数再入栈,最后pc入栈入栈完之后,想知道参数的个数就要读取format,但要读取format就得知道参数的个数,陷入了一个死循环。
  4. 如果函数压栈顺序是从右至左,未知个数的参数先入栈,format再入栈,最后压pc入栈。这时候要想知道参数的个数只需要将栈顶指针加2即可读取到format。
  5. 函数调用栈从高往低使用

使用不定参模拟实现printf

int vasprintf(char **strp, const char *fmt, va_list ap);

vasprintf 是一个 C 库函数,它可以通过可变参数创建一个格式化的字符串,并将其存储在动态分配的内存中。它的使用方法与 printf 类似,但它不会将结果打印到标准输出流中,而是将其存储在一个指向字符数组的指针中

void MyPrint(const char *fmt, ...)
{
    va_list ap;
    va_start(ap,fmt);
    char* res;
    int ret=vasprintf(&res,fmt,ap);
    if(ret  !=-1)
    {
        printf(res);
        free(res);
    }
    va_end(ap);
    
}

C++风格不定参数的使用

需要借助C++11内的万能引用与完美转发及可变参数模版包

#include 
#include 
#include 
#include 

//特化
void CgagaPrint()
{
    std::cout<<std::endl;
}

template<class T,class ...Args>
void CgagaPrint(const T& v,Args&& ...args)
{
    std::cout<<v;
    if((sizeof ...(args))>0)
    {
    	//完美转发
        CgagaPrint(std::forward<Args>(args)...);
    }
    else
    {
        CgagaPrint();
    }
}

int main()
{
    CgagaPrint("nihao",6,5,4);
}

你可能感兴趣的:(C++,c语言,c++,开发语言)