对变长参数的访问是通过下面三个宏实现的:
- 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
参考