当前位置: 首页 > article >正文

C语言中的va_list

目录

1. 可变参数函数(Variadic Function)

2. va_list 及相关宏

3. va_list 的用途

4. 与 printf、vsnprintf 等函数的关系

5. 在实际场景中的示例

5.1 API_SendAtCommandParam 函数

5.2 va_arg 直接取参数

6. 常见问题

7. 结论


在 C 语言中,va_list 是一个专门用于处理可变参数函数(variadic function)的类型。可变参数函数指的是像 printf(const char *format, ...) 这样使用省略号(...)来接收不定数量或类型参数的函数。要想在函数内部获取这些“不定参数”,就要用到 va_list 及其相关的宏和函数。下面做详细介绍:


1. 可变参数函数(Variadic Function)

在 C 语言中,函数可以声明成类似下面的形式:

int exampleFunction(const char *format, ...);

这里的 ... 表示这个函数可以接收除固定参数 format 之外的任意数量任意类型的额外参数(C 语言本身不会自动帮你识别它们的类型,需手动解析)。

一个典型示例就是标准库的 printf 函数原型:

int printf(const char *format, ...);

它可以根据 format 字符串中的占位符(如 %d, %s, %f等)来解析后续传入的参数。


2. va_list 及相关宏

C 语言提供以下专门的机制用来处理可变参数:

  1. va_list

    • 它是一种类型,用来声明一个存储可变参数信息的变量。例如:
      va_list args;
      
  2. va_start(va_list, last_fixed_param)

    • 用来初始化这个 va_list 变量,让它从函数形参中“最后一个固定参数”后面开始读取不定参数。
    • 例如:
      void exampleFunction(const char *format, ...)
      {
          va_list args;
          va_start(args, format); // 初始化args,从format后面获取可变参数
          ...
      }
      
  3. va_arg(va_list, type)

    • 用来按指定类型依次获取下一个可变参数值。例如:
      int i = va_arg(args, int);      // 取一个 int
      double d = va_arg(args, double);// 取一个 double
      char *s = va_arg(args, char*); // 取一个字符串指针
      
  4. va_end(va_list)

    • 用来清理可变参数列表。当函数对不定参数读取完毕后,应当调用 va_end(args)
    • 例如:
      va_end(args);
      
  5. va_copy(va_list dest, va_list src) (C99 引入)

    • 用来复制一个可变参数列表(在某些情形需要重复遍历参数时使用)。

3. va_list 的用途

在一个可变参数函数内部,当你想要处理形如 (const char *fmt, ...) 中的 “..." 部分,就必须:

  1. 声明一个 va_list 变量
  2. va_start(args, last_fixed_param) 初始化它
  3. 根据需要多次调用 va_arg(args, type) 获取后续的每个不定参数
  4. 在处理完后,调用 va_end(args) 进行清理

例如,在一个简化的“打印函数”中:

#include <stdarg.h>  // 包含va_list等定义
#include <stdio.h>

void simplePrint(const char *format, ...)
{
    va_list args;
    va_start(args, format);    // 初始化args,从format后面取不定参数

    // 假设我们只想取一个 int 和一个 double
    int i = va_arg(args, int); 
    double d = va_arg(args, double);

    // 这里可以使用 i, d
    printf("format=%s, i=%d, d=%f\n", format, i, d);

    va_end(args);
}

当调用 simplePrint("test", 42, 3.14);时,va_arg(args, int) 会得到 42,va_arg(args, double) 会得到 3.14。


4. 与 printfvsnprintf 等函数的关系

标准库的 printffprintfvsnprintf 等都使用了可变参数机制:

  • printf(const char *format, ...) / snprintf(char *buf, size_t size, const char *format, ...)
    这类函数使用省略号接收不定参数,然后在函数内部完成对这些参数的解析与格式化输出。

  • vprintf(const char *format, va_list args) / vsnprintf(char *buf, size_t size, const char *format, va_list args)
    针对已有 va_list 的版本。如果你已经拿到了 va_list args,就可以使用 vprintf/vsnprintf 直接处理。
    也就是说,“v”系列函数(如 vprintfvsnprintf)正是可变参数 + va_list 相结合的结果。


5. 在实际场景中的示例

5.1 API_SendAtCommandParam 函数

就像我们之前给出的示例:

#include <stdarg.h>
#include <stdio.h>

API_Status API_SendAtCommandParam(const char *at_template, ...)
{
    char cmd[128];
    va_list args;
    va_start(args, at_template);            // 初始化args,从at_template之后开始获取可变参

    int n = vsnprintf(cmd, sizeof(cmd), at_template, args); 
    va_end(args);

    if (n < 0 || n >= (int)sizeof(cmd)) {
        return API_PARAM_ERROR;
    }

    if (comSendBuf(cmd, n) != 0) {
        return API_ERROR;
    }
    return API_OK;
}

在这里:

  1. 我们的函数定义了一个不定参数 (const char *at_template, ...)
  2. 函数内部声明 va_list args;
  3. 调用 va_start(args, at_template); 来初始化 args
  4. 然后把 args 交给 vsnprintf 来格式化到 cmd 中。
  5. 最后 va_end(args); 进行清理。

5.2 va_arg 直接取参数

如果我们不想使用 vsnprintf 之类函数来拼接字符串,而是自行解析每个参数,可以像这样:

void exampleFunc(const char *info, ...)
{
    va_list args;
    va_start(args, info);

    int a = va_arg(args, int);
    char c = (char) va_arg(args, int); // char在va_arg中传递时要按int取
    double d = va_arg(args, double);

    // ...
    va_end(args);
}

6. 常见问题

  1. 为什么要 va_end

    • 这是可变参数的协议要求,用于清理资源或使处理器栈保持一致。忽略可能导致移植性问题。
  2. 如果不知道参数数量咋办?

    • 一般通过format字符串或某个“终止标识”来知道需要获取多少参数,与 printf 同理。C 语言本身不自动帮你检测参数数量。
  3. va_arg(args, type) 中的 type 必须跟传入参数类型一致吗?

    • 必须一致,否则会导致解析错误、内存访问混乱。
  4. 变长宏(variadic macros)va_list 有何不同?**

    • 变长宏是一种特性(以 ... 结尾的宏定义)在编译期处理多余参数。
    • va_list函数层面的不定参数在运行期处理。

7. 结论

  • va_list 是 C 语言用来处理不定参数函数的一种类型,它与 va_start / va_arg / va_end 等宏配合使用,能够在函数里逐个获取传入的可变数量/类型参数。
  • 典型用法:在函数内部先 va_start(args, last_fixed_param),然后多次 va_arg(args, type) 依次读出每个参数,最后 va_end(args)
  • printfsnprintfvsnprintf 等函数就是使用可变参数(或 va_list)来实现的,在变长参数的解析格式化方面非常常见。

http://www.kler.cn/a/464448.html

相关文章:

  • 算法题(25):只出现一次的数字(三)
  • 海南省大数据发展中心:数据资产场景化评估案例手册(第二期)
  • CG顶会论文阅读|《科技论文写作》硕士课程报告
  • 【机器学习】【朴素贝叶斯分类器】从理论到实践:朴素贝叶斯分类器在垃圾短信过滤中的应用
  • unity学习5:创建一个自己的3D项目
  • 日常工作常用命令集合
  • 云架构Web端的工业MES系统设计之区分工业过程
  • 工业路由器是什么?ER5000为何是领先5G路由器行业
  • 鸿蒙HarmonyOS开发:系统服务(拨打电话、网络搜索、联系人、位置服务、拉起弹框请求用户授权)
  • OpenCV报错:应用程序无法正常启动0xc000007b
  • Hack The Box-Starting Point系列Responder
  • CSS列表、表格、鼠标、滤镜样式设置
  • 深入理解 C 语言预处理:从源文件到可执行程序的关键步骤
  • Vue3实战教程》24:Vue3自定义指令
  • linux下安装达梦数据库v8详解
  • 通过Dockerfile来实现项目可以指定读取不同环境的yml包
  • 24.Java 新特性扩展(重复注解、类型注解)
  • Docker隔离及资源限制原理
  • 参观华为-拓宽全球视野
  • ip属地是看运营商吗还是手机
  • 【C语言 采集数据 精简排序】
  • 数字化转型 · OCR 技术如何打破效率瓶颈?
  • SpringMVC(六)拦截器
  • 栈及栈的操作
  • 【three.js】材质(Material)
  • 《探寻真正开源的大模型:开启AI创新新纪元》