溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務(wù)條款》

如何在C語言中使用可變參數(shù)

發(fā)布時間:2021-01-08 14:40:50 來源:億速云 閱讀:183 作者:Leah 欄目:開發(fā)技術(shù)

這期內(nèi)容當(dāng)中小編將會給大家?guī)碛嘘P(guān)如何在C語言中使用可變參數(shù),文章內(nèi)容豐富且以專業(yè)的角度為大家分析和敘述,閱讀完這篇文章希望大家可以有所收獲。

一,簡單實例

int simple(int num,...)
{
  int i, result=0;
  va_list vl;  //va_list指針,用于va_start取可變參數(shù),為char*
  va_start(vl,num);  //取得可變參數(shù)列表中的第一個值
  printf("num:%d, vl:%d\n",num,*vl);
  for (i = 0; i < (num-1); i++)//這里num表示可變參數(shù)列表中有多少個參數(shù) 
 {
  result = va_arg(vl, int);//這里把vl往后跳過4個字節(jié)(sizeof(int)大?。┲赶蛳乱粋€參數(shù),返回的是當(dāng)前參數(shù)(而非下
一個參數(shù)) 
  printf("in for result:%d, *vl:%d\n", result, *vl);//這里打印下,可以看出,vl總是指向result后面的那個參數(shù) 
 }
 va_end(vl);//結(jié)束標(biāo)志 

 return result;

}
int main(int argc, char **argv)
{
  int num = argc;
  int i = 0;
  simple(5,1,2,3,4,5);

  return 1;

}

運(yùn)行結(jié)果如下:

book@book-desktop:~/own$ ./varlist
num:5, vl:1
in for result:1, *vl:2
in for result:2, *vl:3
in for result:3, *vl:4
in for result:4, *vl:5

二.相關(guān)API介紹

可變參數(shù)列表的實現(xiàn)是由幾個宏組成的,在文件include/stdarg.h中:

va_list  定義某個變量,內(nèi)核中的定義:

  typedef char *va_list;//字符指針類型

va_start(ap, type)   開始獲取可變參數(shù)列表中的第一個參數(shù)(...里面的第一個),也就是跳過第一個參數(shù)(即num):

#ifndef __sparc__
#define va_start(AP, LASTARG)             \
 (AP = ((char *) &(LASTARG) + __va_rounded_size (LASTARG)))//ap指向下一個參數(shù),lastarg不變
#else
#define va_start(AP, LASTARG)             \
 (__builtin_saveregs (),            \
 AP = ((char *) &(LASTARG) + __va_rounded_size (LASTARG))) //跳過下第一個參數(shù),指向第二個參數(shù)內(nèi)存地址
#endif

//對type向上取整 取int的整 4,然后乘上int整型4的倍數(shù)
#define __va_rounded_size(TYPE) \
 (((sizeof (TYPE) + sizeof (int) - 1) / sizeof (int)) * sizeof (int))

va_arg(args, int)  循環(huán)獲取到可變參數(shù)列表中的參數(shù),args指向下一個參數(shù)地址,返回的則是當(dāng)前參數(shù)地址

// first=va_arg(args,int)
#define va_arg(AP, TYPE)            \//ap指向下一個類型的參數(shù)
 (AP += __va_rounded_size (TYPE),          \//返回ap - sizeof(type)參數(shù),即前一個參數(shù)
 *((TYPE *) (AP - __va_rounded_size (TYPE))))

//對type向上取整 取int的整 4,然后乘上int整型4的倍數(shù)
#define __va_rounded_size(TYPE) \
 (((sizeof (TYPE) + sizeof (int) - 1) / sizeof (int)) * sizeof (int))

最后一個va_end(ap)結(jié)束標(biāo)志,可能只是在程序中作為一個可變參數(shù)列表的結(jié)束標(biāo)志而已(stdarg.h里面只是僅僅定義了下,沒有實現(xiàn)的代碼部分)。

三.可變參事應(yīng)用是注意事項

因為va_start, va_arg, va_end等定義成宏,所以它顯得很愚蠢,可變參數(shù)的類型和個數(shù)完全在該函數(shù)中由程序代碼控制,它并不能智能地識別不同參數(shù)的個數(shù)和類型.有人會問:那么printf中不是實現(xiàn)了智能識別參數(shù)嗎?那是因為函數(shù)printf是從固定參數(shù)format字符串來分析出參數(shù)的類型,再調(diào)用va_arg的來獲取可變參數(shù)的.也就是說,你想實現(xiàn)智能識別可變參數(shù)的話是要通過在自己的程序里作判斷來實現(xiàn)的.另外有一個問題,因為編譯器對可變參數(shù)的函數(shù)的原型檢查不夠嚴(yán)格,對編程查錯不利.

如將simple可變參數(shù)該成char型指針,若存在空指針在會產(chǎn)生coredump

void simple(int i, ...) 
{ 
  va_list arg_ptr; 
  char *s=NULL; 

  va_start(arg_ptr, i); 
  s=va_arg(arg_ptr, char*); 
  va_end(arg_ptr); 
  printf("%d %s\n", i, s); 
  return; 
}

可變參數(shù)為char*型,當(dāng)我們忘記用兩個參數(shù)來調(diào)用該函數(shù)時,就會出現(xiàn)core dump(Unix) 或者頁面非法的錯誤(window平臺).但也有可能不出錯,但錯誤卻是難以發(fā)現(xiàn),不利于我們寫出高質(zhì)量的程序.

上述就是小編為大家分享的如何在C語言中使用可變參數(shù)了,如果剛好有類似的疑惑,不妨參照上述分析進(jìn)行理解。如果想知道更多相關(guān)知識,歡迎關(guān)注億速云行業(yè)資訊頻道。

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進(jìn)行舉報,并提供相關(guān)證據(jù),一經(jīng)查實,將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI