va_arg
来自cppreference.com
| 在标头 <cstdarg> 定义
|
||
| |
||
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) |
制造可变函数实参的副本 (宏函数) |
| 结束对可变函数实参的遍历 (宏函数) | |
va_arg 的 C 文档
| |