C语言利用va_list、va_start、va_end、va_arg宏定义可变参数的函数

Wesley13
• 阅读 613

在定义可变参数的函数之前,先来理解一下函数参数的传递原理:

1、函数参数是以栈这种数据结构来存取的,在函数参数列表中,从右至左依次入栈。

2、参数的内存存放格式:参数的内存地址存放在内存的堆栈段中,在执行函数的时候,从最后一个(最右边)参数开始入栈。因此栈底高地址,栈顶低地址,举个例子说明一下:

void test(int a, float b, char c);

那么,在调用test函数的时候,实参char c先进栈,然后是float b,最后才是int a,因此在内存中变量的存放次序是c->b->a,因为从理念上来说,我们只要探测到任意一个变量的地址,并且知道其它变量的类型,通过指针移位运算,就可以顺藤摸瓜找到其它的输入变量。

实现一个可变参数的函数,需要用到以下几个宏:

typedef char* va_list;    // 用于声明一个指向参数列表的字符型指针变量
void va_start(va_list ap, prev_param);    // 第一个参数为指向可变参数字符指针变量,第二个参数是可变参数的第一个参数,通常用于指定可变参数列表中参数的个数
void va_arg(va_list ap, type);    // 第一个参数为指向可变参数字符指针变量,第二个参数是可变参数的数据类型
void va_end(va_list ap);// 将存放可变参数字符串的变量清空(赋值为NULL)

3、示例:求N个数的和

int sum(int count, ...)
{
    int sum = 0;
    int i;
    va_list ap;
    va_start(ap, count);
    for (i = 0; i < count; ++i)
    {
        sum += va_arg(ap, int);
    }
    va_end(ap);
    return sum;
}

int main(int argc, const char * argv[])
{
    int ret = sum(5, 1, 2, 3, 4, 5);
    printf("sum: %d\n",ret);
}

4、 示例:标准库可变参数的相关函数使用

void test(int count,...)
{
    va_list ap;
    va_start(ap, count);
    vprintf("%d,%d,%d\n", ap);                  // 格式化输出可变参数的值
    
    char buff[1024];
    vsprintf(buff, "a=%d,b=%d,c=%d\n", ap);     // 将可变参数列表的值格式化输出到缓冲区中
    printf("%s\n",buff);
    
    vfprintf(stdout, "a=%d,b=%d,c=%d\n", ap);   // 将可变参数列表的值打印到标准输出中
    
    // 可变参数传入的是int类型的地址
    vsscanf("10,30,40", "%d,%d,%d", ap);        // 依次将格式化字符串的值,输入到可变参数变量中
    
    vsnprintf(buff, 30, "a=%d,b=%d,c=%d", ap);  // 将可变参数列表的值,格式化输出指定长度(30个字符)的字符串到缓冲区中
    printf("vsnprintf=%s\n",buff);
    
    va_end(ap);
}
点赞
收藏
评论区
推荐文章
blmius blmius
2年前
MySQL:[Err] 1292 - Incorrect datetime value: ‘0000-00-00 00:00:00‘ for column ‘CREATE_TIME‘ at row 1
文章目录问题用navicat导入数据时,报错:原因这是因为当前的MySQL不支持datetime为0的情况。解决修改sql\mode:sql\mode:SQLMode定义了MySQL应支持的SQL语法、数据校验等,这样可以更容易地在不同的环境中使用MySQL。全局s
Irene181 Irene181
2年前
浅析Python函数的参数
一、前言Python的函数定义非常简单,但灵活度却非常大。除了正常定义的必选参数外,还可以使用默认参数、可变参数和关键字参数,使得函数定义出来的接口,不但能处理复杂的参数,还可以简化调用者的代码。二、可变参数在Python函数中,还可以定义可变参数。顾名思义,可变参数就是传入的参数个数是可变的,可以是1个、2个到任意个,还可以是0个。案例:请计算a2
Jacquelyn38 Jacquelyn38
2年前
2020年前端实用代码段,为你的工作保驾护航
有空的时候,自己总结了几个代码段,在开发中也经常使用,谢谢。1、使用解构获取json数据let jsonData  id: 1,status: "OK",data: 'a', 'b';let  id, status, data: number   jsonData;console.log(id, status, number )
皕杰报表之UUID
​在我们用皕杰报表工具设计填报报表时,如何在新增行里自动增加id呢?能新增整数排序id吗?目前可以在新增行里自动增加id,但只能用uuid函数增加UUID编码,不能新增整数排序id。uuid函数说明:获取一个UUID,可以在填报表中用来创建数据ID语法:uuid()或uuid(sep)参数说明:sep布尔值,生成的uuid中是否包含分隔符'',缺省为
Stella981 Stella981
2年前
KVM调整cpu和内存
一.修改kvm虚拟机的配置1、virsheditcentos7找到“memory”和“vcpu”标签,将<namecentos7</name<uuid2220a6d1a36a4fbb8523e078b3dfe795</uuid
Wesley13 Wesley13
2年前
mysql中时间比较的实现
MySql中时间比较的实现unix\_timestamp()unix\_timestamp函数可以接受一个参数,也可以不使用参数。它的返回值是一个无符号的整数。不使用参数,它返回自1970年1月1日0时0分0秒到现在所经过的秒数,如果使用参数,参数的类型为时间类型或者时间类型的字符串表示,则是从1970010100:00:0
Easter79 Easter79
2年前
Twitter的分布式自增ID算法snowflake (Java版)
概述分布式系统中,有一些需要使用全局唯一ID的场景,这种时候为了防止ID冲突可以使用36位的UUID,但是UUID有一些缺点,首先他相对比较长,另外UUID一般是无序的。有些时候我们希望能使用一种简单一些的ID,并且希望ID能够按照时间有序生成。而twitter的snowflake解决了这种需求,最初Twitter把存储系统从MySQL迁移
Stella981 Stella981
2年前
JS 对象数组Array 根据对象object key的值排序sort,很风骚哦
有个js对象数组varary\{id:1,name:"b"},{id:2,name:"b"}\需求是根据name或者id的值来排序,这里有个风骚的函数函数定义:function keysrt(key,desc) {  return function(a,b){    return desc ? ~~(ak
Wesley13 Wesley13
2年前
12、ES6形参默认值
当定义函数的时候,可以给参数设置默认值。调用的时候不传递参数值,就使用默认值。例子1:普通函数,不传递参数值。默认全是undefined。functionadd(a,b,c){console.log(a,b,c);}add();//不传递参数是,默认参数值全是undef
Stella981 Stella981
2年前
HIVE 时间操作函数
日期函数UNIX时间戳转日期函数: from\_unixtime语法:   from\_unixtime(bigint unixtime\, string format\)返回值: string说明: 转化UNIX时间戳(从19700101 00:00:00 UTC到指定时间的秒数)到当前时区的时间格式举例:hive   selec