|
本文主要介紹va_start和va_end的使用及原理。 在以前的一篇帖子Format MessageBox 詳解中曾使用到va_start和va_end這兩個宏,但對它們也只是泛泛的了解。 介紹這兩個宏之前先看一下C中傳遞函數(shù)的參數(shù)時的用法和原理: 1.在C中,當我們無法列出傳遞函數(shù)的所有實參的類型和數(shù)目時,可以用省略號指定參數(shù)表 void foo(...); void foo(parm_list,...); 這種方式和我們以前認識的不大一樣,但我們要記住這是C中一種傳參的形式,在后面我們就會用到它。 2.函數(shù)參數(shù)的傳遞原理 函數(shù)參數(shù)是以數(shù)據(jù)結(jié)構(gòu):棧的形式存取,從右至左入棧。 首先是參數(shù)的內(nèi)存存放格式:參數(shù)存放在內(nèi)存的堆棧段中,在執(zhí)行函數(shù)的時候,從最后一個開始入棧。因此棧底高地址,棧頂?shù)偷刂?,舉個例子如下: void func(int x, float y, char z); 那么,調(diào)用函數(shù)的時候,實參 char z 先進棧,然后是 float y,最后是 int x,因此在內(nèi)存中變量的存放次序是 x->y->z,因此,從理論上說,我們只要探測到任意一個變量的地址,并且知道其他變量的類型,通過指針移位運算,則總可以順藤摸瓜找到其他的輸入變量。 下面是 <stdarg.h> 里面重要的幾個宏定義如下: 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 是一個字符指針,可以理解為指向當前參數(shù)的一個指針,取參必須通過這個指針進行。 <Step 1> 在調(diào)用參數(shù)表之前,定義一個 va_list 類型的變量,(假設va_list 類型變量被定義為ap); <Step 2> 然后應該對ap 進行初始化,讓它指向可變參數(shù)表里面的第一個參數(shù),這是通過 va_start 來實現(xiàn)的,第一個參數(shù)是 ap 本身,第二個參數(shù)是在變參表前面緊挨著的一個變量,即“...”之前的那個參數(shù); <Step 3> 然后是獲取參數(shù),調(diào)用va_arg,它的第一個參數(shù)是ap,第二個參數(shù)是要獲取的參數(shù)的指定類型,然后返回這個指定類型的值,并且把 ap 的位置指向變參表的下一個變量位置; <Step 4> 獲取所有的參數(shù)之后,我們有必要將這個 ap 指針關(guān)掉,以免發(fā)生危險,方法是調(diào)用 va_end,他是輸入的參數(shù) ap 置為 NULL,應該養(yǎng)成獲取完參數(shù)表之后關(guān)閉指針的習慣。說白了,就是讓我們的程序具有健壯性。通常va_start和va_end是成對出現(xiàn)。 例如 int max(int n, ...); 其函數(shù)內(nèi)部應該如此實現(xiàn): temp ;#include <iostream.h> void fun(int a, ...) { int *temp = &a; for (int i = 0; i < a; i) Output:: 3:獲取省略號指定的參數(shù) 在函數(shù)體中聲明一個va_list,然后用va_start函數(shù)來獲取參數(shù)列表中的參數(shù),使用完畢后調(diào)用va_end()結(jié)束。像這段代碼: void TestFun(char* pszDest, int DestLen, const char* pszFormat, ...) { va_list args; va_start(args, pszFormat); //一定要“...”之前的那個參數(shù) _vsnprintf(pszDest, DestLen, pszFormat, args); va_end(args); } 4.演示如何使用參數(shù)個數(shù)可變的函數(shù),采用ANSI標準形式 #include 〈stdio.h〉 #include 〈string.h〉 #include 〈stdarg.h〉 /*函數(shù)原型聲明,至少需要一個確定的參數(shù),注意括號內(nèi)的省略號*/ int demo( char, ... ); void main( void ) { demo("DEMO", "This", "is", "a", "demo!", ""); } /*ANSI標準形式的聲明方式,括號內(nèi)的省略號表示可選參數(shù)*/ int demo( char msg, ... ) { /*定義保存函數(shù)參數(shù)的結(jié)構(gòu)*/ va_list argp; int argno = 0; char para; /*argp指向傳入的第一個可選參數(shù),msg是最后一個確定的參數(shù)*/ va_start( argp, msg ); while (1) { para = va_arg( argp, char); if ( strcmp( para, "") == 0 ) break; printf("Parameter #%d is: %s\n", argno, para); argno ; } va_end( argp ); /*將argp置為NULL*/ return 0; } 以上是對va_start和va_end的介紹。
最后,希望轉(zhuǎn)載的朋友能夠尊重作者的勞動成果,加上轉(zhuǎn)載地址:http://www.cnblogs.com/hanyonglu/archive/2011/05/07/2039916.html 謝謝。
完畢。^_^
|
|
|
來自: SamBookshelf > 《C 》