文章目錄
一、基礎部分1.1 什麼是可變長參數
可變長參數:顧名思義,就是函數的參數長度(數量)是可變的。比如 C 語言的 printf 系列的(格式化輸入輸出等)函數,都是參數可變的。下面是 printf 函數的聲明:
int printf ( const char * format, ... );
可變參數函式宣告方式都是類似的。
1.2 如何?
C語言可變參數通過三個宏(va_start、va_end、va_arg)和一個類型(va_list)實現的,
void va_start ( va_list ap, paramN );
參數:
ap: 可變參數列表地址
paramN: 確定的參數
功能:初始化可變參數列表(把函數在 paramN 之後的參數地址放到 ap 中)。
void va_end ( va_list ap );
功能:關閉初始化列表(將 ap 置空)。
type va_arg ( va_list ap, type );
功能:返回下一個參數的值。
va_list :儲存參數的類型資訊。
好了,綜合上面3個宏和一個類型可以猜出如何?C語言可變長參數函數:用 va_start 擷取參數列表(的地址)儲存到 ap 中,用 va_arg 逐個擷取值,最後用 va_arg 將 ap 置空。
舉例:
#include <stdio.h>#include <stdarg.h>#define END -1int va_sum (int first_num, ...){ // (1) 定義參數列表 va_list ap; // (2) 初始化參數列表 va_start(ap, first_num); int result = first_num; int temp = 0; // 擷取參數值 while ((temp = va_arg(ap, int)) != END) { result += temp; } // 關閉參數列表 va_end(ap); return result;}int main (){ int sum_val = va_sum(1, 2, 3, 4, 5, END); printf ("%d", sum_val); return 0;}二、知識擴充可能大家也猜到了,我擴充要擴充什麼了?!^_^
簡單介紹兩種函數呼叫慣例__stdcall (C++預設)
- 參數從右向左壓入堆棧
- 函數被調用者修改堆棧
- 函數名(在編譯器這個層次)自動加前置的底線,後面緊跟一個@符號,其後緊跟著參數的尺寸
__cdecl (C語言預設)
- 參數從右向左壓入堆棧
- 參數由調用者清楚,手動清棧,被調用函數不會要求調用者傳遞多少參數,調用者傳遞過多或者過少的參數,甚至完全不同的參數都不會產生編譯階段的錯誤。
那麼,變參函數的調用方式為(也只能是):__cdecl 。
本來打算多寫一點擴充的,又擔心會文不符題,所以感興趣的朋友可以去看參考資料中的文章,有一些介紹的很詳細。