服务器之家:专注于VPS、云服务器配置技术及软件下载分享
分类导航

PHP教程|ASP.NET教程|Java教程|ASP教程|编程技术|正则表达式|C/C++|IOS|C#|Swift|Android|VB|R语言|JavaScript|易语言|vb.net|

服务器之家 - 编程语言 - C/C++ - C 语言变长参数及其陷阱

C 语言变长参数及其陷阱

2023-12-05 13:57coding日记 C/C++

使用 C 风格的变长参数列表访问参数并不安全。这种方法存在几个风险,从 printInts() 函数可以看出。

C 语言变长参数及其陷阱

C 工具

变长参数列表

这部分解释了旧的 C 风格变长参数列表。了解这些内容很重要,因为你可能会在遗留代码中遇到它们。然而,在新代码中,你应该使用变参模板来实现类型安全的变长参数列表。

考虑 C 函数 printf(),来自 。你可以用任意数量的参数调用它:

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++ 提供了语法和一些实用宏,用于编写你自己的变长参数函数。这些函数通常看起来很像 printf()。尽管你不经常需要这个特性,但偶尔你会遇到它相当有用的情况。例如,假设你想编写一个快速而简单的调试函数,当设置了调试标志时,该函数将字符串打印到 stderr,但如果没有设置调试标志,则不执行任何操作。就像 printf() 一样,这个函数应该能够打印具有任意数量和任意类型参数的字符串。一个简单的实现如下:

#include 
#include 

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() 的原型包含一个类型化且命名的参数 str,后面跟着 ...(省略号)。它们代表任意数量和类型的参数。要访问这些参数,你必须使用  中定义的宏。你声明一个 va_list 类型的变量,并用 va_start 调用进行初始化。va_start() 的第二个参数必须是参数列表中最右边的命名变量。所有具有变长参数列表的函数都至少需要一个命名参数。debugOut() 函数简单地将这个列表传递给 vfprintf()( 中的标准函数)。vfprintf() 调用返回后,debugOut() 调用 va_end() 来终止访问变量参数列表。在调用 va_start() 后,你必须始终调用 va_end(),以确保函数以一致的堆栈状态结束。你可以如下方式使用该函数:

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);

访问参数

如果你想自己访问实际参数,你可以使用 va_arg() 来做到这一点。它接受 va_list 作为第一个参数,以及要解释的参数的类型。不幸的是,除非你提供明确的方式,否则无法知道参数列表的结尾。例如,你可以使第一个参数是参数数量的计数。或者,在你有一组指针的情况下,你可能需要最后一个指针是 nullptr。有许多方法,但它们都对程序员来说是繁琐的。

下面的示例演示了调用者在第一个命名参数中指定提供了多少个参数的技术。该函数接受任意数量的 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;
}

你可以按以下方式调用 printInts()。请注意,第一个参数指定将跟随多少个整数。

printInts(5, 5, 4, 3, 2, 1);

为什么不应使用 C 风格的变长参数列表

访问风险

使用 C 风格的变长参数列表访问参数并不安全。这种方法存在几个风险,从 printInts() 函数可以看出:

  • 不知道参数的数量:在 printInts() 的情况下,你必须信任调用者作为第一个参数传递正确数量的参数。在 debugOut() 的情况下,你必须信任调用者在字符数组后传递的参数数量与字符数组中的格式化代码数量相同。
  • 不知道参数的类型:va_arg() 接受一个类型,用它来解释其当前位置的值。然而,你可以告诉 va_arg() 将值解释为任何类型。它无法验证正确的类型。

警告:避免使用 C 风格的变长参数列表。建议传递一个 std::array 或 vector 的值、使用初始化列表,或者使用类型安全的变参模板来实现变长参数列表。

原文地址:https://mp.weixin.qq.com/s?__biz=Mzg2NzY2NTUyMg==&mid=2247484665&idx=1&sn=b1186270bd901bf3b09d22fab069a694

延伸 · 阅读

精彩推荐
  • C/C++C语言读写配置文件的方法

    C语言读写配置文件的方法

    这篇文章主要介绍了C语言读写配置文件的方法,包括C语言读写ini配置文件所涉及的文件读写技巧,以及完整的源文件及头文件实现方法,需要的朋友可以参考...

    华宰7382021-03-03
  • C/C++C++直接初始化与复制初始化的区别深入解析

    C++直接初始化与复制初始化的区别深入解析

    这篇文章主要介绍了C++直接初始化与复制初始化的区别深入解析,是很多C++初学者需要深入了解的重要概念,需要的朋友可以参考下...

    C++教程网5852021-01-30
  • C/C++C语言求逆矩阵案例详解

    C语言求逆矩阵案例详解

    这篇文章主要介绍了C语言求逆矩阵案例详解,本篇文章通过简要的案例,讲解了该项技术的了解与使用,以下就是详细内容,需要的朋友可以参考下...

    dogdng5852021-12-16
  • C/C++C语言文字艺术之数据输入输出

    C语言文字艺术之数据输入输出

    这篇文章主要介绍了C语言文字艺术之数据输入输出,C语言的语句用来向计算机系统发出操作指令。一条语句编写完成经过编译后产生若干条机器指...

    王小王_1236742023-02-23
  • C/C++c++中的bind使用方法

    c++中的bind使用方法

    bind是这样一种机制,它可以预先把指定可调用实体的某些参数绑定到已有的变量,产生一个新的可调用实体,这种机制在回调函数的使用过程中也颇为有用...

    xutopia6522022-08-10
  • C/C++C++中Boost.Chrono时间库的使用方法

    C++中Boost.Chrono时间库的使用方法

    chrono是一个time library, 源于boost,现在已经是C++11标准了,下面这篇文章主要给大家介绍了关于C++中Boost.Chrono时间库的使用方法,文中通过示例代码介绍的非常...

    taozj6692021-05-31
  • C/C++C语言实现删除某一个数组值的方法

    C语言实现删除某一个数组值的方法

    这篇文章主要给大家分享C语言数组中删除数组中某个值的方法,既然要学习删除数组中的元素,我们就必须得先知道数组中有哪些元素。同时还要定义一个...

    不羁4882022-02-24
  • C/C++C++ 开发之实现操作符重载的实例

    C++ 开发之实现操作符重载的实例

    这篇文章主要介绍了C++ 开发之实现操作符重载的实例的相关资料,这里附有实例代码和实现效果图帮助大家参考实践,需要的朋友可以参考下...

    liuyi12071643399182021-05-24