VA_LIST是在C語言中解決變參問題的一組宏,所在頭文件:#include stdarg.h,用于獲取不確定個數(shù)的參數(shù)。
創(chuàng)新互聯(lián)是一家集網(wǎng)站建設(shè),廉江企業(yè)網(wǎng)站建設(shè),廉江品牌網(wǎng)站建設(shè),網(wǎng)站定制,廉江網(wǎng)站建設(shè)報價,網(wǎng)絡(luò)營銷,網(wǎng)絡(luò)優(yōu)化,廉江網(wǎng)站推廣為一體的創(chuàng)新建站企業(yè),幫助傳統(tǒng)企業(yè)提升企業(yè)形象加強(qiáng)企業(yè)競爭力??沙浞譂M足這一群體相比中小企業(yè)更為豐富、高端、多元的互聯(lián)網(wǎng)需求。同時我們時刻保持專業(yè)、時尚、前沿,時刻以成就客戶成長自我,堅(jiān)持不斷學(xué)習(xí)、思考、沉淀、凈化自己,讓我們?yōu)楦嗟钠髽I(yè)打造出實(shí)用型網(wǎng)站。
VA_LIST的用法:
首先在函數(shù)里定義一具VA_LIST型的變量,這個變量是指向參數(shù)的指針;
然后用VA_START宏初始化剛定義的VA_LIST變量;
然后用VA_ARG返回可變的參數(shù),VA_ARG的第二個參數(shù)是你要返回的參數(shù)的類型(如果函數(shù)有多個可變參數(shù)的,依次調(diào)用VA_ARG獲取各個參數(shù));
最后用VA_END宏結(jié)束可變參數(shù)的獲取。
以下是一個定義一個參數(shù)個數(shù)不確定的函數(shù)的簡單例子:
#include?cstdarg
#include?iostream
using?namespace?std;
double?average?(?int?num,?...?)
{
va_list?arguments;?????????????????????//?A?place?to?store?the?list?of?arguments
double?sum?=?0;
va_start?(?arguments,?num?);???????????//?Initializing?arguments?to?store?all?values?after?num
for?(?int?x?=?0;?x??num;?x++?)????????//?Loop?until?all?numbers?are?added
sum?+=?va_arg?(?arguments,?double?);?//?Adds?the?next?value?in?argument?list?to?sum.
va_end?(?arguments?);??????????????????//?Cleans?up?the?list
return?sum?/?num;??????????????????????//?Returns?some?number?(typecast?prevents?truncation)
}
int?main()
{
cout?average?(?3,?12.2,?22.3,?4.5?)?endl;
cout?average?(?5,?3.3,?2.2,?1.1,?5.5,?3.3?)?endl;
}
在函數(shù)定義的時候就必須寫形參名了。聲明的時候不需要,因?yàn)榫幾g器還用不到,只要知道相應(yīng)位置是個啥類型的就行了
首先先看到main函數(shù)中的不定參數(shù):
1.引用:在Turbo C2.0啟動過程中, 傳遞main()函數(shù)三個參數(shù): argc, argv和env。
* argc: 整數(shù), 鴐ain()的命令行參數(shù)個數(shù)。
* argv: 字符串?dāng)?shù)組。
argv[0] 為程序運(yùn)行的全路徑名
argv[1] 為在DOS命令行中執(zhí)行程序名后的第一個字符串;
argv[2] 為執(zhí)行程序名后的第二個字符串;
...
argv[argc]為NULL。
*env: 字符串?dāng)?shù)組。env[] 的每一個元素都包含ENVVAR=value形式的字符
串。其中ENVVAR為環(huán)境變量。value 為ENVVAR的對應(yīng)值
#include stdlib.h
#include stdio.h
main(int argc, char *argv[], char *env[])
{
int i;
printf("%d\n", argc); /* 為什么它的輸出是 1, 它到底是定義什么的,我看不明上面的解釋*/
for(i=0; i=argc; i++)
printf("argv[%d]:%s\n", i, argv[i]);
for(i=0; env[i]!=NULL; i++)
printf(" env[%d]:%s\n", i, env[i]);
}
argc, argv, env是在main()函數(shù)之前被賦值的,編譯器生成的可執(zhí)行文件,main()不是真正的入口點(diǎn),而是一個標(biāo)準(zhǔn)的函數(shù),這個函數(shù)名與具體的操作系統(tǒng)有關(guān)。
就想到其他函數(shù)是否能實(shí)現(xiàn)一樣的功能,查詢了相關(guān)資料,基本上都是利用STDARG.H中的
#define va_start(ap, parmN) (ap = ...)
#define va_arg(ap, type) (*((type *)(ap))++)
#define va_end(ap)
定義如下:
typedef char * va_list;
#define va_start _crt_va_start
#define va_arg _crt_va_arg
#define va_end _crt_va_end
#define _crt_va_start(ap,v) ( ap = (va_list)_ADDRESSOF(v) + _INTSIZEOF(v) )
#define _crt_va_arg(ap,t) ( *(t *)((ap += _INTSIZEOF(t)) - _INTSIZEOF(t)) )
#define _crt_va_end(ap) ( ap = (va_list)0 )
修改他人程序如下:
void average(int first,...)
{int i=first;
va_list maker;
va_start(maker,first);
while(i!=-1){ printf("%p: %d\n",maker,i);
i=va_arg(maker,int);
}
}
void main(void)
{
average(2,3,4,4,-1);
}
運(yùn)行結(jié)果:
FFCE: 2
FFD0: 3
FFD2: 4
FFD4: 4
這個程序顯示函數(shù)參數(shù)的地址相差2個字節(jié)
所以可以改寫為:
void x(char *n,...)
{int *p;
p=n;
while(*p!=-1)
{printf("%p:%s\n",p,*p,*p);
p+=sizeof(char);}}
void main()
{x("g","hfd","gfg","vsds",-1);}
運(yùn)行結(jié)果:
FFD2:g
FFD4:hfd
FFD6:gfg
FFD8:vsds
不必通過宏va_start,va_arg,va_end實(shí)現(xiàn)
C語言中定義了va_list, va_start( ), va_arg( ), va_end( ) 這樣一組宏來處理可變參數(shù)問題。
可以參考printf的聲明,創(chuàng)建自己的實(shí)現(xiàn)函數(shù)。示例:
#includestdio.h
#includestdarg.h
void?va_fun(int?start,...)
{
va_list?arg_ptr;
int?nArgValue?=?start;
int?nArgCount?=?1;
va_start(arg_ptr,start);
while(nArgValue?!=?-1)
{
printf("arg?%d?is:%d/n",nArgCount,nArgValue);
nArgValue=va_arg(arg_ptr,int);
++nArgCount;?
}
return;?
}
main()
{
va_fun(5,1,7,-1);
printf("................/n");
va_fun(2,4,-1);
printf("................/n");
va_fun(-1);
printf("................/n");
//va_fun();?可變參數(shù)函數(shù)最少要有一個參數(shù)。
}