本文主要介紹va_start和va_end的使用及原理。
在以前的一篇帖子format messagebox 詳解
中曾使用到va_start和va_end這兩個巨集,但對它們也只是泛泛的了解。
介紹這兩個巨集之前先看一下c中傳遞函式的引數時的用法和原理:
1.在c中,當我們無法列出傳遞函式的所有實參的型別和數目時,可以用省略號指定參數列
void foo(...);
void foo(parm_list,...);
這種方式和我們以前認識的不大一樣,但我們要記住這是c中一種傳參的形式,在後面我們就會用到它。
2.函式引數的傳遞原理
函式引數是以資料結構:棧的形式訪問,從右至左入棧。
首先是引數的記憶體存放格式:引數存放在記憶體的堆疊段中,在執行函式的時候,從最後乙個開始入棧。因此棧底高位址,棧頂低位址,舉個例子如下:
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,應該養成獲取完參數列之後關閉指標的習慣。說白了,就是讓我們的程式具有健壯性。通常va_start和va_end是成對出現。
例如 int max(int n, ...); 其函式內部應該如此實現:
#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.演示如何使用引數個數可變的函式,採用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; }
以上是對va_start和va_end的介紹。
va start和va end使用詳解
本文主要介紹va start和va end的使用及原理。在以前的一篇帖子format messagebox 詳解 中曾使用到va start和va end這兩個巨集,但對它們也只是泛泛的了解。介紹這兩個巨集之前先看一下c中傳遞函式的引數時的用法和原理 1.在c中,當我們無法列出傳遞函式的所有實參的型...
va start和va end的使用
1 由於在c語言中沒有函式過載,解決不定數目函式引數問題變得比較麻煩 即使採用c 如果引數個數不能確定,也很難採用函式過載.對這種情況,有些人採用指標引數來解決問題.下面就c語言中處理不定引數數目的問題進行討論.2 下面是 裡面重要的幾個巨集定義如下 typedef char va list voi...
va start和va end使用詳解
介紹這兩個巨集之前先看一下c中傳遞函式的引數時的用法和原理 1.在c中,當我們無法列出傳遞函式的所有實參的型別和數目時,可以用省略號指定參數列 void foo void foo parm list,這種方式和我們以前認識的不大一樣,但我們要記住這是c中一種傳參的形式,在後面我們就會用到它。2.函式...