在C中,可变参数用于参数个数,类型不确定的情况,如printf,snprintf函数的实现。
当我们无法列出传递函数的所有实参的类型和数目时,可以用省略号指定参数表
void func(...);
void func(parm_list,...);
这是C传参的一种形式,与固定参数不同。
函数参数的传递原理
函数参数以栈的形式存储,从右往左入栈。
举个例子:
void func(int x, float y, char z);
在调用函数的时候,实参z先入栈,然后是y,最后是x。在内存中变量的存放次序是x->y->z。所以,从理论上来说,只要知道x,y,z其中一个变量的地址和类型,通过指针运算,可找到其他变量。
typedef char* va_list;
void va_start ( va_list ap, prev_param ); /* ANSI version */
type va_arg ( va_list ap, type );
void va_end ( va_list ap );
va_list是指向当前参数的指针,通过这个指针进行取参。
宏的使用方式如下:
- 先定义一个va_list的变量,假设为ap
- 使用va_start对ap进行初始化,va_start的第一个参数是ap,第二个参数是变参表中“…”前面的那个参数
- 然后调用va_arg获取参数,第一个参数还是ap,第二个参数是要获取的参数的类型,并且ap指向下一个变量
- 获取完参数后,使用va_end关掉ap。va_start和va_end通常成对出现。
例子:
#include <stdio.h>
#include <stdlib.h>
#include <stdarg.h>
int my_snprintf(char *dst, int size, char *fmt, ...)
{
int len;
va_list argp;
va_start(argp, fmt);
len = vsnprintf(dst, size, fmt, argp);
len = len > size - 1 ? size - 1 : len;
va_end(argp);
return len;
}
int main(void)
{
char str[8];
int len;
len = my_snprintf(str, sizeof(str), "A:%d:%s", 1, "ABCDEFGH");
printf("str:%s, len:%d\n", str, len);
return 0;
}