C++可變參數的實現方法

來源:互聯網
上載者:User

可變參數的實現要解決三個問題:

1.如何調用帶有可變參數的函數
2.如何編譯有可變參數的程式
3.在帶有可變參數的函數體中如何持有可變參數
第一個問題, 調用時在可以傳入可變參數的地方傳入可變參數即可,當然,還有一些需要注意的地方,後面會提到。

第二個問題,編譯器需要在編譯時間採用一種寬鬆的檢查方案,,這會帶來一些問題, 比如對編程查錯不利。

第三個是我在這裡要關心的問題,先以C語言為例分析其實現原理。

printf和scanf是C語言標準庫中最常見的可變參數函數, printf的簽名是

複製代碼 代碼如下:int printf(const char* format, ...);

其中,... 表示可變參數,現在模仿printf寫一個簡單的例子。

一、一個簡單了例子:

複製代碼 代碼如下:#include <windows.h>
#include <stdio.h>

void VariableArgumentMethod(int argc, ...);

int main(){
VariableArgumentMethod(6, 4, 7, 3, 0, 7, 9);
return 0;
}

void VariableArgumentMethod(int argc, ...){
// 聲明一個指標, 用於持有可變參數
va_list pArg;
// 將 pArg 初始化為指向第一個參數
va_start(pArg, argc);
// 輸出參數
  for(int i = 0; i != argc; ++i){
// 擷取 pArg 所指向的參數並輸出
printf("%d, ", va_arg(pArg, int) );
}

va_end(pArg);
}

void VariableArgumentMethod(int argc, ...)是一個可變參數函數,這個函數用於將 argc 指定個數的可變參數輸出。
VariableArgumentMethod(6, 4, 7, 3, 0, 7, 9); 是對這個函數的調用,第一個實參 6 表示後面跟了 6 個參數。

在 VariableArgumentMethod 的函數體中:

1. va_list pArg;

定義了一個用於持有可變參數的指標,通過將這個指標在傳入的可變參數表中移動,可以持有第一個可變參數。

2. va_start(pArg, argc);

讓 pArg 指向可變參數列表中的第一個參數。argc 是一個用來定位的參數,因為可變參數是從 argc 後開始的,後面會說明為什麼要這樣定位。

3. va_arg(pArg, int);

這句話放在迴圈體中,用於取出可變參數表中的參數。並且,它會讓 pArg 移向下個可變參數(如果已經到達末尾,則它將指向一個沒有意義的地址)。

4. va_end(pArg);

給 pArg 清零,個人認為在這裡可有可無,因為 pArg 已經不需要了。

就這樣,VariableArgumentMethod 函數體遍曆了可變參數表中傳入的參數,並用printf("%d, ", va_arg(pArg, int) ) 進行了輸出。

二、實現細節

1. 先瞭解一下編譯器如何處理傳遞參數這個問題的。

編譯器是將參數壓入棧中進行傳遞的。傳遞實參的時候,編譯器會從實參列表中,按從右至左的順序將參數入棧,對於 VariableArgumentMethod(6, 4, 7, 3, 0, 7, 9)調用,則入棧的順序是 9, 7, 0, 3, 7, 4, 6 (注意沒有可變參數與不可變參數之分)。由於棧的地址是從高到低的,所以實參入棧後,實參在棧中的分布如。可以看出,實參在棧中,還是保持了左邊參數處於低地址,右邊參數處於高地址的狀態。OK,知道這些就夠了。

低地址                            高地址

...

6

4

7

3

0

7

9

...

2. va_list, va_start, va_arg 和 va_end

va_list 是一個定義的指標類型,va_start, va_arg 和 va_end 都是C語言用於處理可變參數而定義的宏,在stdarg.h檔案中。由於硬體平台的不同,編譯器的不同,導致它們的定義也有所不同,但基本思路相同。以下是相關宏的定義。

複製代碼 代碼如下:typedef char * va_list;

#define _ADDRESSOF(v) ( &(v) )

#define _INTSIZEOF(n) ( (sizeof(n) + sizeof(int) - 1) & ~(sizeof(int) - 1) )

#define va_start(ap,v) ( ap = (va_list)_ADDRESSOF(v) + _INTSIZEOF(v) )

#define va_arg(ap,t) ( *(t *)((ap += _INTSIZEOF(t)) - _INTSIZEOF(t)) )

#define va_end(ap) ( ap = (va_list)0 )

可以看出,此處引入了另外兩個宏 _ADDRESSOF 和 _INTSIZEOF。

_ADDRESSOF(v) 是用於擷取變數地址的,這一眼就能看出來;

_INTSIZEOF(n) 是用於對齊的。(什麼是對齊呢?這是因為棧的結構導致的,在 32 位機中,棧中每個單元都是占 4 個位元組的,這往往是一個 int 型的長度,但實際傳過來的參數可能並不正好是 4 個位元組,或者正好是 4 的倍數個位元組,就好像坐車時不會賣半個座位給乘客一樣,如果傳入的資料沒有正好占 4 個或 4 的倍數個位元組,則需要對齊(補齊)。至於為什麼這個運算式能夠對齊,需要分析一下);

va_start(ap,v) 中,ap 是用於持有可變參數的指標, v 是最後一個非可變參數的參數,(va_list)_ADDRESSOF(v) 擷取 v 的地址,並轉為 va_list 類型的,v 是最後一個非可變參數的參數,在本例中應為 6, 在中處理棧的低地址端,_INTSIZEOF(v) 擷取了一個對齊地址,這裡應為 4, 兩個相加後,即指向了第一個可變參數,即中的 4, 將這個值賦給 ap 後,就讓 ap 指向了第一個可變參數。(從這裡可以看出,將va_list 定義為 char* 是很有用的,因為 char 長度為一個位元組,便於指標運算);

va_arg(ap,t) 中,ap 是用於持有可變參數的指標,t 是要擷取參數的類型,ap += _INTSIZEOF(t) 讓 ap 指向下一個參數,但是,此處還需要擷取當前參數的值,所以又將運算式減回來,返回的應是一個 va_list(char*) 型的指標,因此要轉型為 t* 後再進行解引用運算,得到當前參數的值。(注意這裡有個將 ap 移向下一個參數又減回來的操作,本人感覺不太好,一方面這裡有個浪費的操作,對效能會有一些影響,另一方面,我更希望將取當前值的操作和移向下一個的操作分離,這樣可以讓程式員有更多的控制,並且容易理解。)

va_end(ap) 則是讓 ap 指向一個空地址。

通過以上分析,可以發現,C 語言中可變參數是從棧中按順序訪問的,過程中所使用的三個宏,也只是對操作的簡單封裝,完全可以自己編程實現。而且,參數的類型和個數是不能直接確定的,在本例中,VariableArgumentMethod 的第一個參數用於指定參數的個數,而參數的類型約定為整形,這樣程式才能正常運行,再說到 printf,它之所以能識別參數的個數,是因為它的第一個參數中必須要描述後面參數的格式字串,這正是一開始所提到的第一個問題中說到的要注意的問題。這也是它被很多人所詬病的原因,但是,本人認為這種方式是很好的,後面會與 java 和 .net 的實現方式進行比較。

三、java 和 .net 實現可變參數的方式。

java 從1.5以後,開始支援可變參數,其定義文法為:

複製代碼 代碼如下:void testMethod(String ... args)

對於這個方法,可以這樣調用:testMethod("gly", "zxy", "ChenFei");

.net 也支援可變參數,其定義文法為:

複製代碼 代碼如下:void TestMethod(params string[] args)

對於這個方法,可以這樣調用:TestMethod("gly", "zxy", "ChenFei");

在 java 和 .net 中,對於可變參數的實現基本是一樣的:編譯器在編譯時間,將方法簽名中的可變參數視為相應類型的數組,編譯相應的調用時,根據實參產生一個數組,將參數裝入到數組中進行傳遞,而在可變參數方法的方法體中,按使用數組的方式使用可變參數。

四、兩種實現方式的比較

C 語言的實現方式與 java .net 的實現方式相比,C 語言需要程式員做更多的工作,而且,確實增加了出錯的機會,java .net 的實現方式可以很容易的確定參數的類型和個數,這些 C 的實現中是沒有的,但是 java .net 的實現方式會產生臨時數組,當然 java .net 有記憶體回收機制,但是,垃圾什麼時候被回收是不確定的,而且是代價很大的,記憶體回收是個好東西,但我不喜歡,我認為不需要的東西應該立即釋放,這是完美的一個方面的體現。C 中沒有這個問題,參數的個數和類型問題可以靠約定或指定來解決,而這兩個問題在 java 和 .net 中,參數個數其實是間接傳遞過去了(數組的長度),參數類型則是在方法簽名中約定了。當然,java .net 的設計目標和 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.