va start va end 的使用和原理

2021-06-02 01:15:03 字數 2846 閱讀 7474

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 先介紹一下可變參數列的呼叫形式以及原理 首先是引數的記憶體存放格式 引數存放在記憶體的堆疊段中,...