關(guān)于C/C++中可變參數(shù)的詳細(xì)介紹(va_list,va_start,va_arg,va_end)
由于在C語(yǔ)言中沒(méi)有函數(shù)重載,解決不定數(shù)目函數(shù)參數(shù)問(wèn)題變得比較麻煩,即使采用C++,如果參數(shù)個(gè)數(shù)不能確定,也很難采用函數(shù)重載。對(duì)這種情況,提出了指針參數(shù)來(lái)解決問(wèn)題。
如printf()函數(shù),其原型為:
int printf( const char* format, ...);
它除了有一個(gè)參數(shù)format固定以外,后面跟的參數(shù)的個(gè)數(shù)和類型是可變的,例如我們可以有以下不同的調(diào)用方法:
printf( "%d ",i);
printf( "%s ",s);
printf( "the number is %d ,string is:%s ", i, s);
如何實(shí)現(xiàn)其功能?
我們需要以下幾個(gè)宏定義:
(1)va_list
定義了一個(gè)指針arg_ptr, 用于指示可選的參數(shù).
(2)va_start(arg_ptr, argN)
使參數(shù)列表指針arg_ptr指向函數(shù)參數(shù)列表中的第一個(gè)可選參數(shù),argN是位于第一個(gè)可選參數(shù)之前的固定參數(shù), 或者說(shuō)最后一個(gè)固定參數(shù).如有一va函數(shù)的聲明是void va_test(char a, char b, char c, ...), 則它的固定參數(shù)依次是a,b,c, 最后一個(gè)固定參數(shù)argN為c, 因此就是va_start(arg_ptr, c).
(3)va_arg(arg_ptr, type)
返回參數(shù)列表中指針arg_ptr所指的參數(shù), 返回類型為type. 并使指針arg_ptr指向參數(shù)列表中下一個(gè)參數(shù).返回的是可選參數(shù), 不包括固定參數(shù).
(4)va_end(arg_ptr)
清空參數(shù)列表, 并置參數(shù)指針arg_ptr無(wú)效.
(注:va在這里是variable-argument(可變參數(shù))的意思. 這些宏定義在stdarg.h中,所以用到可變參數(shù)的程序應(yīng)該包含這個(gè)頭文件)
也需你現(xiàn)在還是不能理解,別著急,現(xiàn)在從一個(gè)實(shí)例著手.定義這么一個(gè)函數(shù),函數(shù)的第一個(gè)參數(shù)是固定的,其余參數(shù)是可變的。定義為:
void simple_va_fun(int i,...); 其代碼為:
#include <iostream>
#include <stdarg.h>
using namespace std;
void simple_va_fun(int i,...);
int main(int argc,char *argv[])
{
simple_va_fun(100);
simple_va_fun(100,200);
simple_va_fun(100,200,'a');
return 0;
}
void simple_va_fun(int i,...)
{
va_list arg_ptr; //定義可變參數(shù)指針
va_start(arg_ptr,i); // i為最后一個(gè)固定參數(shù)
int j=va_arg(arg_ptr,int); //返回第一個(gè)可變參數(shù),類型為int
char c=va_arg(arg_ptr,char); //返回第二個(gè)可變參數(shù),類型為char
va_end(arg_ptr); // 清空參數(shù)指針
printf( "%d %d %c\n",i,j,c);
return;
}
代碼運(yùn)行解釋:
(1)首先在函數(shù)里定義一個(gè)va_list型的變量,這里是arg_ptr,這個(gè)變量是指向參數(shù)的指針.
(2)然后用va_start宏初始化變量arg_ptr,這個(gè)宏的第二個(gè)參數(shù)是第一個(gè)可變參數(shù)的前一個(gè)參數(shù),是一個(gè)固定的參數(shù).
(3)然后用va_arg返回第一個(gè)可變的參數(shù),并賦值給整數(shù)j。va_arg的第二個(gè)參數(shù)是你要返回的參數(shù)的類型,這里是int型. 返回第一個(gè)可變參數(shù)后arg_ptr指向第二個(gè)可變參數(shù),用同樣的方法返回并賦值給c,類型為char類型。
(4)最后用va_end宏結(jié)束可變參數(shù)的獲取。
小結(jié):
可變參數(shù)的函數(shù)原理其實(shí)很簡(jiǎn)單,而va系列是以宏定義來(lái)定義的,實(shí)現(xiàn)跟堆棧相關(guān).我們寫(xiě)一個(gè)可變函數(shù)的C函數(shù)時(shí),有利也有弊,所以在不必要的場(chǎng)合,我們無(wú)需用到可變參數(shù).如果在C++里,我們應(yīng)該利用C++的多態(tài)性來(lái)實(shí)現(xiàn)可變參數(shù)的功能,盡量避免用C語(yǔ)言的方式來(lái)實(shí)現(xiàn)。
附加:
參數(shù)在堆棧中分布:
在進(jìn)程中,堆棧地址是從高到低分配的.當(dāng)執(zhí)行一個(gè)函數(shù)的時(shí)候,將參數(shù)列表入棧,壓入堆棧的高地址部分,然后入棧函數(shù)的返回地址,接著入棧函數(shù)的執(zhí)行代碼,這個(gè)入棧過(guò)程,堆棧地址不斷遞減,一些黑客就是在堆棧中修改函數(shù)返回地址,執(zhí)行自己的代碼來(lái)達(dá)到執(zhí)行自己插入的代碼段的目的. 總之,函數(shù)在堆棧中的分布情況是:地址從高到低,依次是:函數(shù)參數(shù)列表,函數(shù)返回地址,函數(shù)執(zhí)行代碼段. 堆棧中,各個(gè)函數(shù)的分布情況是倒序的.即最后一個(gè)參數(shù)在列表中地址最高部分,第一個(gè)參數(shù)在列表地址的最低部分.參數(shù)在堆棧中的分布情況如下:
最后一個(gè)參數(shù)
倒數(shù)第二個(gè)參數(shù)
...
第一個(gè)參數(shù)
函數(shù)返回地址
函數(shù)代碼段
相關(guān)文章
C++標(biāo)準(zhǔn)之(ravalue reference) 右值引用介紹
臨時(shí)對(duì)象的產(chǎn)生和拷貝所帶來(lái)的效率折損,一直是C++所為人詬病的問(wèn)題,下面簡(jiǎn)單地介紹一下Copy Elision、RVO,對(duì)此不感興趣的可以直接跳過(guò)2012-11-11
C語(yǔ)言進(jìn)階教程之循環(huán)語(yǔ)句缺陷詳析
循環(huán)語(yǔ)句是用于重復(fù)執(zhí)行某條語(yǔ)句(循環(huán)體)的語(yǔ)句,它包含一個(gè)控制表達(dá)式,每循環(huán)執(zhí)行一次都要對(duì)控制表達(dá)式進(jìn)行判斷,如果表達(dá)式為真,則繼續(xù)執(zhí)行循環(huán),這篇文章主要給大家介紹了關(guān)于C語(yǔ)言進(jìn)階教程之循環(huán)語(yǔ)句缺陷的相關(guān)資料,需要的朋友可以參考下2021-08-08
C語(yǔ)言中操作進(jìn)程信號(hào)的相關(guān)函數(shù)使用詳解
這篇文章主要介紹了C語(yǔ)言中操作進(jìn)程信號(hào)的相關(guān)函數(shù)使用詳解,分別是signal()函數(shù)和kill()函數(shù)的用法,需要的朋友可以參考下2015-09-09
Linux下動(dòng)靜態(tài)庫(kù)的打包與使用指南(C/C++)
c++是面向?qū)ο蟮木幊陶Z(yǔ)言,比較方便實(shí)現(xiàn)某些第三方庫(kù),比如翻譯其他面向?qū)ο笳Z(yǔ)言的代碼,比c語(yǔ)言要方便的多,下面這篇文章主要給大家介紹了關(guān)于Linux下C/C++動(dòng)靜態(tài)庫(kù)的打包與使用的相關(guān)資料,需要的朋友可以參考下2023-02-02
C語(yǔ)言實(shí)現(xiàn)linux網(wǎng)卡連接檢測(cè)的方法
這篇文章主要為大家詳細(xì)介紹了C語(yǔ)言實(shí)現(xiàn)linux網(wǎng)卡連接檢測(cè)的方法,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2018-06-06
C++實(shí)現(xiàn)設(shè)計(jì)模式之裝飾者模式詳解
這篇文章主要介紹了C++設(shè)計(jì)模式之裝飾模式,裝飾模式能夠?qū)崿F(xiàn)動(dòng)態(tài)的為對(duì)象添加功能,是從一個(gè)對(duì)象外部來(lái)給對(duì)象添加功能,需要的朋友可以參考下2021-09-09
OpenCV中C++函數(shù)imread讀取圖片的問(wèn)題及解決方法
利用C++函數(shù)imread讀取圖片的時(shí)候返回的結(jié)果總是空,而利用C函數(shù)cvLoadImage時(shí)卻能讀取到圖像。怎么回事?今天小編通過(guò)本教程給大家簡(jiǎn)單說(shuō)明原因2017-03-03
matlab遺傳算法求解車間調(diào)度問(wèn)題分析及實(shí)現(xiàn)源碼
這篇文章主要為大家介紹了matlab遺傳算法求解車間調(diào)度問(wèn)題解析,文中附含詳細(xì)實(shí)現(xiàn)源碼,有需要的朋友可以借鑒參考下,希望能夠有所幫助2022-02-02

