va_arg
来自cppreference.com
在标头 <cstdarg> 定义
|
||
T va_arg( std::va_list ap, T ); |
||
va_arg
宏展开成对应于 va_list ap 的下个参数的 T
类型的表达式。
在调用 va_arg
前,ap 必须用对 va_start 或 va_copy 的调用初始化,中间不能夹带 va_end 的调用。va_arg
宏的每次调用都修改 ap,令它指向下个可变实参。
若 ap 中的下个实参的类型(提升后)与 T 不兼容,则行为未定义,除非:
- 一个类型是有符号整数类型,而另一类型是无符号整数类型,且其值能用两个类型表示;或
- 一个类型是指向 void 的指针,而另一类型是指向字符类型(char、signed char 或 unsigned char)的指针。
若在 ap 中没有更多实参时调用 va_arg
,则行为未定义。
参数
ap | - | va_list 类型的实例 |
T | - | ap 中下个参数的类型 |
展开值
ap 中的下个可变参数
示例
运行此代码
#include <cstdarg> #include <cstdio> #include <iostream> void print_variance(std::size_t count, const char* fmt, ...) { double sum = 0; double sum_sq = 0; std::va_list args; va_start(args, fmt); for (std::size_t i = count; i--;) { double num = va_arg(args, double); sum += num; sum_sq += num*num; } va_end(args); std::printf(fmt, sum_sq / count - (sum / count) * (sum / count)); } void nano_printf(const char* fmt, ...) { std::va_list args; va_start(args, fmt); for (const char* p = fmt; *p != '\0'; ++p) { switch (*p) { case '%': switch (*++p) // 读取格式符号 { case 'i': std::cout << va_arg(args, int); continue; case 'f': std::cout << va_arg(args, double); continue; case 's': std::cout << va_arg(args, const char*); continue; case 'c': std::cout << static_cast<char>(va_arg(args, int)); continue; case '%': std::cout << '%'; continue; /* ...更多情况... */ } break; // 格式错误... case '\n': std::cout << '\n'; continue; case '\t': std::cout << '\t'; continue; /* ...更多情况... */ } std::cout << *p; } va_end(args); } int main() { print_variance(4, "%f\n", 25.0, 27.3, 26.9, 25.7); nano_printf("Args: %i%% %c%f %s\n", 42, '#', 3.14, "C++"); }
输出:
0.846875 Args: 42% #3.14 C++
参阅
启用对可变函数实参的访问 (宏函数) | |
(C++11) |
制造可变函数实参的副本 (宏函数) |
结束对可变函数实参的遍历 (宏函数) |