當我們無法列出傳遞函數的所有實參的類型和數目時,可以用省略號指定參數表
- (void)parameterList:(NSString *)a,...;
函數參數是以數據結構:棧的形式存取,從右至左入棧。
void func(int x, float y, char z);
調用函數的時候,實參 char z 先進棧,然后是 float y,最后是 int x,因此在內存中變量的存放次序是 x->y->z,因此,從理論上說,我們只要探測到任意一個變量的地址,并且知道其他變量的類型,通過指針移位運算,則總可以找到其他的輸入變量。
獲取參數:
//va_list 是一個字符指針
//可以理解為指向當前參數的一個指針
//取參必須通過這個指針進行。
typedef char * va_list;
void va_start ( va_list ap, prev_param );
type va_arg ( va_list ap, type );
void va_end ( va_list ap );
- 在調用參數表之前,定義一個 va_list 類型的變量,(假設va_list 類型變量被定義為ap);
- 然后應該對ap 進行初始化,讓它指向可變參數表里面的第一個參數,這是通過 va_start 來實現的,第一個參數是 ap 本身,第二個參數是在變參表前面緊挨著的一個變量,即“...”之前的那個參數;
- 然后是獲取參數,調用va_arg,它的第一個參數是ap,第二個參數是要獲取的參數的指定類型,然后返回這個指定類型的值,并且把 ap 的位置指向變參表的下一個變量位置;
- 獲取所有的參數之后,我們有必要將這個 ap 指針關掉,以免發生危險,方法是調用 va_end,他是輸入的參數 ap 置為 NULL。通常va_start和va_end是成對出現。
例子:
- (void)parameterList:(int)a,... {
va_list argp; //定義保存函數參數的結構
int argno = 0;
int para; //argp指向傳入的第一個可選參數,a是最后一個確定的參數
va_start( argp, a);
while (1) {
para = va_arg( argp, int);
if ( para == 0 ) { break; }
NSLog(@"Parameter #%d is: %d\n", argno, para);
argno++;
}
va_end( argp ); //將argp置為NULL
}
- (void)parameterStringList:(NSString *)msg,...{
va_list argp;
NSString * parameter;
va_start(argp, msg);
while (1) {
parameter = va_arg(argp, NSString *);
if (parameter == nil) { break; }
NSLog(@"parameter: %@", parameter);
}
va_end(argp);
}
運行結果:
可變參數函數調用
References:
http://www.cnblogs.com/hanyonglu/archive/2011/05/07/2039916.html