C 語言變長參數(shù)及其陷阱
C 工具
變長參數(shù)列表
這部分解釋了舊的 C 風(fēng)格變長參數(shù)列表。了解這些內(nèi)容很重要,因為你可能會在遺留代碼中遇到它們。然而,在新代碼中,你應(yīng)該使用變參模板來實現(xiàn)類型安全的變長參數(shù)列表。
考慮 C 函數(shù) printf(),來自 <cstdio>。你可以用任意數(shù)量的參數(shù)調(diào)用它:
printf("int %d\n", 5);
printf("String %s and int %d\n", "hello", 5);
printf("Many ints: %d, %d, %d, %d, %d\n", 1, 2, 3, 4, 5);
C/C++ 提供了語法和一些實用宏,用于編寫你自己的變長參數(shù)函數(shù)。這些函數(shù)通??雌饋砗芟?nbsp;printf()。盡管你不經(jīng)常需要這個特性,但偶爾你會遇到它相當(dāng)有用的情況。例如,假設(shè)你想編寫一個快速而簡單的調(diào)試函數(shù),當(dāng)設(shè)置了調(diào)試標(biāo)志時,該函數(shù)將字符串打印到 stderr,但如果沒有設(shè)置調(diào)試標(biāo)志,則不執(zhí)行任何操作。就像 printf() 一樣,這個函數(shù)應(yīng)該能夠打印具有任意數(shù)量和任意類型參數(shù)的字符串。一個簡單的實現(xiàn)如下:
#include <cstdio>
#include <cstdarg>
bool debug { false };
void debugOut(const char* str, ...) {
va_list ap;
if (debug) {
va_start(ap, str);
vfprintf(stderr, str, ap);
va_end(ap);
}
}
首先,請注意 debugOut() 的原型包含一個類型化且命名的參數(shù) str,后面跟著 ...(省略號)。它們代表任意數(shù)量和類型的參數(shù)。要訪問這些參數(shù),你必須使用 <cstdarg> 中定義的宏。你聲明一個 va_list 類型的變量,并用 va_start 調(diào)用進(jìn)行初始化。va_start() 的第二個參數(shù)必須是參數(shù)列表中最右邊的命名變量。所有具有變長參數(shù)列表的函數(shù)都至少需要一個命名參數(shù)。debugOut() 函數(shù)簡單地將這個列表傳遞給 vfprintf()(<cstdio> 中的標(biāo)準(zhǔn)函數(shù))。vfprintf() 調(diào)用返回后,debugOut() 調(diào)用 va_end() 來終止訪問變量參數(shù)列表。在調(diào)用 va_start() 后,你必須始終調(diào)用 va_end(),以確保函數(shù)以一致的堆棧狀態(tài)結(jié)束。你可以如下方式使用該函數(shù):
debug = true;
debugOut("int %d\n", 5);
debugOut("String %s and int %d\n", "hello", 5);
debugOut("Many ints: %d, %d, %d, %d, %d\n", 1, 2, 3, 4, 5);
訪問參數(shù)
如果你想自己訪問實際參數(shù),你可以使用 va_arg() 來做到這一點。它接受 va_list 作為第一個參數(shù),以及要解釋的參數(shù)的類型。不幸的是,除非你提供明確的方式,否則無法知道參數(shù)列表的結(jié)尾。例如,你可以使第一個參數(shù)是參數(shù)數(shù)量的計數(shù)。或者,在你有一組指針的情況下,你可能需要最后一個指針是 nullptr。有許多方法,但它們都對程序員來說是繁瑣的。
下面的示例演示了調(diào)用者在第一個命名參數(shù)中指定提供了多少個參數(shù)的技術(shù)。該函數(shù)接受任意數(shù)量的 int 并打印出來:
void printInts(size_t num, ...) {
va_list ap;
va_start(ap, num);
for (size_t i { 0 }; i < num; ++i) {
int temp { va_arg(ap, int) };
cout << temp << " ";
}
va_end(ap);
cout << endl;
}
你可以按以下方式調(diào)用 printInts()。請注意,第一個參數(shù)指定將跟隨多少個整數(shù)。
printInts(5, 5, 4, 3, 2, 1);
為什么不應(yīng)使用 C 風(fēng)格的變長參數(shù)列表
訪問風(fēng)險
使用 C 風(fēng)格的變長參數(shù)列表訪問參數(shù)并不安全。這種方法存在幾個風(fēng)險,從 printInts() 函數(shù)可以看出:
- 不知道參數(shù)的數(shù)量:在 printInts() 的情況下,你必須信任調(diào)用者作為第一個參數(shù)傳遞正確數(shù)量的參數(shù)。在 debugOut() 的情況下,你必須信任調(diào)用者在字符數(shù)組后傳遞的參數(shù)數(shù)量與字符數(shù)組中的格式化代碼數(shù)量相同。
- 不知道參數(shù)的類型:va_arg() 接受一個類型,用它來解釋其當(dāng)前位置的值。然而,你可以告訴 va_arg() 將值解釋為任何類型。它無法驗證正確的類型。
警告:避免使用 C 風(fēng)格的變長參數(shù)列表。建議傳遞一個 std::array 或 vector 的值、使用初始化列表,或者使用類型安全的變參模板來實現(xiàn)變長參數(shù)列表。