自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

C 語言變長參數(shù)及其陷阱

開發(fā) 前端
使用 C 風(fēng)格的變長參數(shù)列表訪問參數(shù)并不安全。這種方法存在幾個風(fēng)險,從 printInts() 函數(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ù)列表。

責(zé)任編輯:趙寧寧 來源: coding日記
相關(guān)推薦

2022-07-04 14:41:31

Go 語言變長參數(shù)變長參數(shù)函數(shù)

2022-07-03 23:07:48

Go語言參數(shù)

2025-04-07 08:45:00

C 語言數(shù)組

2011-07-14 10:23:33

C語言

2021-07-08 23:53:44

Go語言拷貝

2022-07-28 12:17:36

C語言typedef#define

2017-07-10 08:30:11

Spark UDF變長參數(shù)

2024-09-30 11:08:18

JSON局限性數(shù)據(jù)

2023-06-26 00:03:55

Go語言類型

2021-02-24 15:05:32

C語言數(shù)組代碼

2011-05-13 17:25:34

C

2010-01-11 13:31:03

C++語言

2010-01-11 11:27:25

C++語言

2010-01-12 17:47:31

C++語言

2020-07-27 08:05:56

C++語言后端

2021-01-06 05:29:04

C語言參數(shù)應(yīng)用

2011-05-24 16:58:52

CC++

2022-02-28 11:13:21

數(shù)據(jù)中心自動化數(shù)據(jù)中心網(wǎng)絡(luò)

2024-03-06 13:23:56

Task.RunC#異步陷阱

2024-12-23 06:20:00

點贊
收藏

51CTO技術(shù)棧公眾號