对变长参数的访问是通过下面三个宏实现的:
void va_start( va_list arg_ptr, prev_param ); // (ANSI version) type va_arg( va_list arg_ptr, type ); void va_end( va_list arg_ptr );
参数
type: 待获取参数的数据类型;
arg_ptr: 指向参数列表的 va_list 类型指针(va_list 是 char* 的别名);
prev_param: 第一个可选参数(用 "..." 标识的参数属可选参数)之前, 并与之相邻的参数的名称. 在 ANSI 版本的宏中, 该参数是必须的.
头文件
<stdio.h> 和 <stdarg.h>
说明
1. va_start 使 arg_ptr 指向传递给函数可选参数列表中的第一个参数;
2. va_start 必须在第一次使用 va_arg 前使用;
3. va_arg 从 arg_ptr 指向的位置获取一个 type 类型的值. 调用结束后, arg_ptr 偏移 sizeof(type) 个字节指向下一个参数的位置. 注意, 在 VS2008withSP1 中, arg_ptr 偏移量大于等于 4. 当 sizeof(type) < 4 时, arg_ptr 偏移 4 个字节;
4. 取完所有参数后, 调用 va_end 可将 arg_ptr 置为 NULL.
代码
// Build with VS2008withSP1, /W4 #include <cstdio> #include <cstdarg> void foo(char arg, char* arg2, ...) { printf_s("sizeof(double) = %d/r/n", sizeof(double)); printf_s("sizeof(unsigned char) = %d/r/n", sizeof(unsigned char)); printf_s("sizeof(int) = %d/r/n", sizeof(int)); printf_s("sizeof(short) = %d/r/n", sizeof(short)); printf_s("sizeof(long long) = %d/r/n", sizeof(long long)); printf_s("/r/n"); printf_s("arg = %c/r/n", arg); printf_s("arg2 = %s/r/n", arg2); va_list arg_ptr; // ! va_start(arg_ptr, arg2); // ! // arg_ptr: 0x003ffba0 double arg3 = va_arg(arg_ptr, double); printf_s("arg3 = %f/r/n", arg3); // arg_ptr: 0x003ffba8 == 0x003ffba0 + sizeof(double) unsigned char arg4 = va_arg(arg_ptr, unsigned char); printf_s("arg4 = %c/r/n", arg4); // arg_ptr: 0x003ffbac != 0x003ffba8 + sizeof(unsigned char) // 0x003ffbac == 0x003ffba8 + 4 int arg5 = va_arg(arg_ptr, int); printf_s("arg5 = %d/r/n", arg5); // arg_ptr: 0x003ffbb0 != 0x003ffbac + sizeof(int) short arg6 = va_arg(arg_ptr, short); printf_s("arg6 = %hd/r/n", arg6); // arg_ptr: 0x003ffbb4 != 0x003ffbb0 + sizeof(short) // 0x003ffbb4 == 0x003ffbb0 + 4 long long arg7 = va_arg(arg_ptr, long long); printf_s("arg7 = %lld/r/n", arg7); // arg_ptr: 0x003ffbbc == 0x003ffbb4 + sizeof(long long) va_end(arg_ptr); // ! // arg_ptr: 0x00000000 } int main() { foo('A', "ABCDEF", 3.14, 'B', 2147483647, 32767, 9223372036854775807); return 0; }
输出
sizeof(double) = 8 sizeof(unsigned char) = 1 sizeof(int) = 4 sizeof(short) = 2 sizeof(long long) = 8 arg = A arg2 = ABCDEF arg3 = 3.140000 arg4 = B arg5 = 2147483647 arg6 = 32767 arg7 = 9223372036854775807
参考
http://msdn.microsoft.com/zh-cn/library/kb57fad8.aspx