一、什麼是可變參數
我們在C語言編程中有時會遇到一些參數個數可變的函數,例如printf()函數,其函數原型為:
- int printf( const char* format, ...);
它除了有一個參數format固定以外,後面跟的參數的個數和類型是可變的(用三個點“…”做參數占位符),實際調用時可以有以下的形式:
- printf("%d",i);
- printf("%s",s);
- printf("the number is %d ,string is:%s", i, s);
以上這些東西已為大家所熟悉。但是究竟如何寫可變參數的C函數以及這些可變參數的函數編譯器是如何實現,這個問題卻一直困擾了我好久。本文就這個問題進行一些探討,希望能對大家有些幫助.
二、寫一個簡單的可變參數的C函數
先看例子程序。該函數至少有一個整數參數,其後是占位符…,表示後面參數的個數不定. 在這個例子裡,所有的輸入參數必須都是整數,函數的功能是打印所有參數的值.
函數代碼如下:
- //示例代碼1:可變參數函數的使用
- #include "stdio.h"
- #include "stdarg.h"
- void simple_va_fun(int start, ...)
- {
- va_list arg_ptr;
- int nArgValue =start;
- int nArgCout=0; //可變參數的數目
- va_start(arg_ptr,start); //以固定參數的地址為起點確定變參的內存起始地址。
- do
- {
- ++nArgCout;
- printf("the %d th arg: %d\n",nArgCout,nArgValue); //輸出各參數的值
- nArgValue = va_arg(arg_ptr,int); //得到下一個可變參數的值
- } while(nArgValue != -1);
- return;
- }
- int main(int argc, char* argv[])
- {
- simple_va_fun(100,-1);
- simple_va_fun(100,200,-1);
- return 0;
- }
從這個函數的實現可以看到,我們使用可變參數應該有以下步驟:
⑴在程序中將用到以下這些宏:
- void va_start( va_list arg_ptr, prev_param );
- type va_arg( va_list arg_ptr, type );
- void va_end( va_list arg_ptr );
va在這裡是variable-argument(可變參數)的意思.
這些宏定義在stdarg.h中,所以用到可變參數的程序應該包含這個頭文件.
⑵函數裡首先定義一個va_list型的變量,這裡是arg_ptr,這個變量是指向參數地址的指針.因為得到參數的地址之後,再結合參數的類型,才能得到參數的值。
⑶然後用va_start宏初始化⑵中定義的變量arg_ptr,這個宏的第二個參數是可變參數列表的前一個參數,也就是最後一個固定參數。
⑷然後依次用va_arg宏使arg_ptr返回可變參數的地址,得到這個地址之後,結合參數的類型,就可以得到參數的值。然後進行輸出。
⑸設定結束條件,這裡的條件就是判斷參數值是否為-1。注意被調的函數在調用時是不知道可變參數的正確數目的,程序員必須自己在代碼中指明結束條件。至於為什麼它不會知道參數的數目,讀者在看完下面這幾個宏的內部實現機制後,自然就會明白。