va_arg
來自 cppreference.com
定義於標頭檔案 <cstdarg> |
||
T va_arg( std::va_list ap, T ); |
||
va_arg
宏擴充套件為型別 T 的表示式,對應於 va_list ap 中的下一個引數。
在呼叫 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) // read format symbol { 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; /* ...more cases... */ } break; // format error... case '\n': std::cout << '\n'; continue; case '\t': std::cout << '\t'; continue; /* ...more cases... */ } 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) |
複製變長函式引數 (函式宏) |
結束變長函式引數的遍歷 (函式宏) | |
C 文件 關於 va_arg
|