1:當無法列出傳遞函式的所有實參的型別和數目時,可用省略號指定參數列
void foo(...);
void foo(parm_list,...);
2:函式引數的傳遞原理
函式引數是以資料結構:棧的形式訪問,從右至左入棧.eg:
先介紹一下可變參數列的呼叫形式以及原理:
首先是引數的記憶體存放格式:引數存放在記憶體的堆疊段中,在執行函式的時候,從最後乙個開始入棧。因此棧底高位址,棧頂低位址,舉個例子如下:
void func(int x, float y, char z);
那麼,呼叫函式的時候,實參 char z 先進棧,然後是 float y,最後是 int x,因此在記憶體中變數的存放次序是 x->y->z,因此,從理論上說,我們只要探測到任意乙個變數的位址,並且知道其他變數的型別,通過指標移位運算,則總可以順藤摸瓜找到其他的輸入變數。
下面是 裡面重要的幾個巨集定義如下:
typedef char* va_list;
void va_start ( va_list ap, prev_param ); /* ansi version */
type va_arg ( va_list ap, type );
void va_end ( va_list ap );
va_list 是乙個字元指標,可以理解為指向當前引數的乙個指標,取參必須通過這個指標進行。
在呼叫參數列之前,定義乙個 va_list 型別的變數,(假設va_list 型別變數被定義為ap);
然後應該對ap 進行初始化,讓它指向可變參數列裡面的第乙個引數,這是通過 va_start 來實現的,第乙個引數是 ap 本身,第二個引數是在變參表前面緊挨著的乙個變數,即「...」之前的那個引數;
然後是獲取引數,呼叫va_arg,它的第乙個引數是ap,第二個引數是要獲取的引數的指定型別,然後返回這個指定型別的值,並且把 ap 的位置指向變參表的下乙個變數位置;
獲取所有的引數之後,我們有必要將這個 ap 指標關掉,以免發生危險,方法是呼叫 va_end,他是輸入的引數 ap 置為 null,應該養成獲取完參數列之後關閉指標的習慣。
例如 int max(int n, ...); 其函式內部應該如此實現:
int max(int n, ...)
va_end(ap); // 善後工作,關閉 ap
return max;
}// 在主函式中測試 max 函式的行為(c++ 格式)
int main()
基本用法闡述至此,可以看到,這個方法存在兩處極嚴重的漏洞:其一,輸入引數的型別隨意性,使得引數很容易以乙個不正確的型別獲取乙個值(譬如輸入乙個float,卻以int型去獲取他),這樣做會出現莫名其妙的執行結果;其二,變參表的大小並不能在執行時獲取,這樣就存在乙個訪問越界的可能性,導致後果嚴重的 runtime error。
#include
void fun(int a, ...) }
int main()
output::
1 2 3 4
3:獲取省略號指定的引數
在函式體中宣告乙個va_list,然後用va_start函式來獲取引數列表中的引數,使用完畢後呼叫va_end()結束。像這段**:
void testfun(char* pszdest, int destlen, const char* pszformat, ...)
4.va_start使argp指向第乙個可選引數。va_arg返回引數列表中的當前引數並使argp指向引數列表中的下乙個引數。va_end把argp指標清為null。函式體內可以多次遍歷這些引數,但是都必須以va_start開始,並以va_end結尾。
1).演示如何使用引數個數可變的函式,採用ansi標準形式
#include 〈stdio.h〉
#include 〈string.h〉
#include 〈stdarg.h〉
/*函式原型宣告,至少需要乙個確定的引數,注意括號內的省略號*/
int demo( char, ... );
void main( void )
/*ansi標準形式的宣告方式,括號內的省略號表示可選引數*/
int demo( char msg, ... )
va_end( argp );
/*將argp置為null*/
return 0; }
2)//示例**1:可變引數函式的使用
#include "stdio.h"
#include "stdarg.h"
void ******_va_fun(int start, ...)
while(nargvalue != -1);
return;
}int main(int argc, char* argv)
3)//示例**2:擴充套件——自己實現簡單的可變引數的函式。
下面是乙個簡單的printf函式的實現,參考了中的例子
#include "stdio.h"
#include "stdlib.h"
void myprintf(char* fmt, ...) //乙個簡單的類似於printf的實現,//引數必須都是int 型別
else
parg += sizeof(int); //等價於原來的va_arg
}++fmt;
}while (*fmt != '/0');
parg = null; //等價於va_end
return;
}int main(int argc, char* argv)
va start va end 的使用和原理
1 當無法列出傳遞函式的所有實參的型別和數目時,可用省略號指定參數列 void foo void foo parm list,2 函式引數的傳遞原理 函式引數是以資料結構 棧的形式訪問,從右至左入棧.eg 先介紹一下可變參數列的呼叫形式以及原理 首先是引數的記憶體存放格式 引數存放在記憶體的堆疊段中...
va start va end 的使用和原理
1 當無法列出傳遞函式的所有實參的型別和數目時,可用省略號指定參數列 void foo void foo parm list,2 函式引數的傳遞原理 函式引數是以資料結構 棧的形式訪問,從右至左入棧.eg 先介紹一下可變參數列的呼叫形式以及原理 首先是引數的記憶體存放格式 引數存放在記憶體的堆疊段中...
va start va end 的使用和原理
1 當無法列出傳遞函式的所有實參的型別和數目時 可用省略號指定參數列 voidfoo void foo parm list,2 函式引數的傳遞原理 函式引數是以資料結構 棧的形式訪問 從右至左入棧 eg 先介紹一下可變參數列的呼叫形式以及原理 首先是引數的記憶體存放格式 引數存放在記憶體的堆疊段中,...