va_arg

来自cppreference.com
< cpp‎ | utility‎ | variadic
 
 
工具库
语言支持
类型支持(基本类型、RTTI)
库功能特性测试宏 (C++20)
动态内存管理
程序工具
协程支持 (C++20)
变参数函数
调试支持
(C++26)
三路比较
(C++20)
(C++20)(C++20)(C++20)
(C++20)(C++20)(C++20)
通用工具
日期和时间
函数对象
格式化库 (C++20)
(C++11)
关系运算符 (C++20 中弃用)
整数比较函数
(C++20)(C++20)(C++20)   
(C++20)
交换类型运算
(C++14)
(C++11)
(C++11)
(C++11)
(C++17)
常用词汇类型
(C++11)
(C++17)
(C++17)
(C++17)
(C++11)
(C++17)
(C++23)
初等字符串转换
(C++17)
(C++17)

 
 
在标头 <cstdarg> 定义
T va_arg( std::va_list ap, T );

va_arg 宏展开成对应于 va_list ap 的下个参数的 T 类型的表达式。

在调用 va_arg 前,ap 必须用对 va_startva_copy 的调用初始化,中间不能夹带 va_end 的调用。va_arg 宏的每次调用都修改 ap,令它指向下个可变实参。

ap 中的下个实参的类型(提升后)与 T 不兼容,则行为未定义,除非:

  • 一个类型是有符号整数类型,而另一类型是无符号整数类型,且其值能用两个类型表示;或
  • 一个类型是指向 void 的指针,而另一类型是指向字符类型(charsigned charunsigned 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)
制造可变函数实参的副本
(宏函数)
结束对可变函数实参的遍历
(宏函数)