C語言中可變參數的用法

來源:互聯網
上載者:User

標籤:arc   實現   單一實例   編譯器   fine   last   int   div   開始   

前言

  在C語言程式編寫中我們使用最多的函數一定包括printf以及很多類似的變形體。這個函數包含在C庫函數中,定義為 int printf( const char* format, ...);

  除了一個格式化字串之外還可以輸入多個可變參量,如:

      printf("%d",i);
      printf("%s",s);
      printf("the number is %d ,string is:%s", i, s);

  格式化字串的判斷本章暫且不論,下面分析一下可變參數的實現細節。

一,簡單一實例

int simple(int num,...){        int i, result=0;        va_list vl;     //va_list指標,用於va_start取可變參數,為char*        va_start(vl,num);       //取得可變參數列表中的第一個值        printf("num:%d, vl:%d\n",num,*vl);        for (i = 0; i < (num-1); i++)//這裡num表示可變參數列表中有多少個參數      {        result = va_arg(vl, int);//這裡把vl往後跳過4個位元組(sizeof(int)大小)指向下一個參數,返回的是當前參數(而非下一個參數)          printf("in for  result:%d,  *vl:%d\n", result, *vl);//這裡列印下,可以看出,vl總是指向result後面的那個參數      }    va_end(vl);//結束標誌      return result;}int main(int argc, char **argv){        int num = argc;        int i = 0;        simple(5,1,2,3,4,5);        return 1;}

  運行結果如下:

[email protected]:~/own$ ./varlist                num:5, vl:1in for  result:1,  *vl:2in for  result:2,  *vl:3in for  result:3,  *vl:4in for  result:4,  *vl:5
二.相關API介紹

  可變參數列表的實現是由幾個宏組成的,在檔案include/stdarg.h中:

  va_list  定義某個變數,核心中的定義:

    typedef char *va_list;//字元指標類型  

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

#ifndef __sparc__#define va_start(AP, LASTARG)                         \ (AP = ((char *) &(LASTARG) + __va_rounded_size (LASTARG)))//ap指向下一個參數,lastarg不變#else#define va_start(AP, LASTARG)                         \ (__builtin_saveregs (),                          AP = ((char *) &(LASTARG) + __va_rounded_size (LASTARG))) //跳過下第一個參數,指向第二個參數記憶體位址#endif//對type向上取整 取int的整 4,然後乘上int整型4的倍數#define __va_rounded_size(TYPE)  \  (((sizeof (TYPE) + sizeof (int) - 1) / sizeof (int)) * sizeof (int))

  va_arg(args, int)  迴圈擷取到可變參數列表中的參數,args指向下一個參數地址,返回的則是當前參數地址

//  first=va_arg(args,int)#define va_arg(AP, TYPE)                        \//ap指向下一個類型的參數 (AP += __va_rounded_size (TYPE),                    \//返回ap - sizeof(type)參數,即前一個參數  *((TYPE *) (AP - __va_rounded_size (TYPE))))//對type向上取整 取int的整 4,然後乘上int整型4的倍數#define __va_rounded_size(TYPE)  \  (((sizeof (TYPE) + sizeof (int) - 1) / sizeof (int)) * sizeof (int))

  最後一個va_end(ap)結束標誌,可能只是在程式中作為一個可變參數列表的結束標誌而已(stdarg.h裡面只是僅僅定義了下,沒有實現的代碼部分)。

三.可變參事應用是注意事項

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

  如將simple可變參數該成char型指標,若存在null 指標在會產生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; }

  可變參數為char*型,當我們忘記用兩個參數來調用該函數時,就會出現core dump(Unix) 或者頁面非法的錯誤(window平台).但也有可能不出錯,但錯誤卻是難以發現,不利於我們寫出高品質的程式.

 

C語言中可變參數的用法

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.