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

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 和一个 doubleint i = va_arg(args, int); double d = va_arg(args, double);// 这里可以使用 i, dprintf("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.lryc.cn/news/515079.html

相关文章:

  • idea无法安装插件
  • 智汇厦门:苏哒智能携其智能化产品亮相文心中国行现场
  • C++函数模板的定义为何要和调用点放在一起
  • Nginx - 整合lua 实现对POST请求的参数拦截校验(不使用Openresty)
  • 互联网直播点播平台EasyDSS无人机视频推拉流技术实现工地远程监控巡检直播
  • Unity3D 基于GraphView实现的节点编辑器框架详解
  • 【C++】开源:Armadillo数值计算库配置与使用
  • HackMyVM-Airbind靶机的测试报告
  • C语言----函数
  • MySQL图形化界面工具--DataGrip
  • PyTorch AMP 混合精度中grad_scaler.py的scale函数解析
  • 【Ubuntu20.04】Apollo10.0 Docker容器部署+常见错误解决
  • 【文献精读笔记】Explainability for Large Language Models: A Survey (大语言模型的可解释性综述)(二)
  • 朱姆沃尔特隐身战舰:从失败到威慑
  • 免费分享 | 基于极光优化算法PLO优化宽度学习BLS实现光伏数据预测算法研究附Matlab代码
  • logback日志文件多环境配置路径
  • 面试高频:一致性hash算法
  • docker部署项目
  • 每天40分玩转Django:Django Celery
  • df.groupby(pd.Grouper(level=1)).sum()
  • 运动控制探针功能详细介绍(CODESYS+SV63N伺服)
  • C语言基础18(GDB调试)
  • 《向量数据库指南》——应对ElasticSearch挑战,拥抱Mlivus Cloud的新时代
  • c++的stl库中stack的解析和模拟实现
  • C语言——字符函数和内存函数
  • 查询docker overlay2文件夹下的 c7ffc13c49xxx是哪一个容器使用的
  • Golang的容器编排实践
  • 【51项目】51单片机自制小霸王游戏机
  • ArkTs之NAPI学习
  • 【数据库初阶】MySQL中表的约束(上)