C语言学习之路--第十七站 可变参数

在函数中定义一个接受可变数量的参数,定义方式为:int func_name(int arg1,...);

在使用时代码抽象如下:

go 复制代码
int func(int, ... )  {
   .
   .
   .
}

int main() {
   func(2, 2, 3);
   func(3, 2, 3, 4);
}
  • 需要使用 stdatg.h 头文件,该文件提供了实现可变参数功能的函数和宏
  • 最后一个参数写成省略号,三个点。 int 表示要传递的可变参数总和。
  • 使用步骤:
    • 定义一个函数,最后参数为省略号,省略号可以设置自定义参数
    • 在函数中创建一个 va_list 类型变量,该类型是在 stdarg.h 头文件中定义的
    • 使用 int 参数和 va_start() 宏来初始化 va_list 变量为一个参数列表。 宏 va_start() 是在 stdarg.h 头文件中定义的
    • 使用 va_arg() 宏 和 va_list 变量来访问参数列表中的每个项
    • 使用宏 va_end() 来清理赋予 va_list 变量的内存
  • 常见的宏使用
    • va_start(ap , last_arg) 初始化可变参数列表。 ap 是 va_list 类型的变量,va_arg 是最后一个固定参数的名称,也可以是可变参数之前的参数。该宏将 ap 指向可变参数列表中的第一个参数
    • va_arg(ap , type) 获取可变参数列表中下一个参数。ap 是va_list 类型的变量,type是下一个参数的变量。该宏返回类型为 type 的值,将 ap 指向下一个参数
    • va_end(ap) 结束可变参数列表的访问。 ap 是va_list 类型的变量,该宏将 ap 设置为 NULL

写一个带有可变参数的函数,返回他们的平均值:

scss 复制代码
#include<stdarg.h>

// 可变参数测试方法
double average(int num, ...)
{
        va_list valist;
        double sum = 0.0;
        int i;

// 为 num 个参数初始化
va_start(valist, num);

//访问所有赋给 valist 的参数
for (i = 0; i < num; i++)
{
	sum += va_arg(valist, int);
}

//清理为 valist 保留的内存
va_end(valist);

return sum / num;
}

printf("Average of 2,3,4,5 = %f \n", average(4, 2, 3, 4, 5));
printf("Average of 10,23,34,2,3,5 = %f\n", average(6,10,23,34,2,3,5));

其实看上面内容还很空洞,并没有理解。下面再参考一些网友的留言。

arduino 复制代码
#include <stdio.h>

void debug_arg(unsigned int num, ...) 
{
    unsigned int i = 0;
    unsigned int *addr = &num;
    for (i = 0; i <= num; i++) 
    {
        /* *(addr + i) 从左往右依次取出传递进来的参数,类似于出栈过程 */
        printf("i=%d,value=%d\r\n", i, *(addr + i));
    }
}
int main(void)
{
    debug_arg(3, 66, 88, 666);
    return 0;
}

下面是源代码, 进行拆分消化一下。

arduino 复制代码
// 64 位机器用 8 字节对齐, 32 位 4 位对齐
#ifdef X64
#defin t long long
#else
#define t int
#endif
  • 判断一下机器类型,64位机器用 8 字节对齐,32位 4 字节对齐。

arduino 复制代码
//VA_LIST套宏中可以使用,用来改变INTSIZEOF中t的类型
//固定参数详见
void test(int a, double b, char* c)
{
    char *p = (char*)&a;
    //因为&a = void 类型 需要转换,void * =&a 不需要转换但是使用时要转换
    printf("%p %p %p\n", &a, &b, &c);
    //观察地址变化
    printf("%p %s",(p+8),*(char**)(p+8+8));//64位机器时加8内存大小8字节对齐
    return;
}

//可变参数实验
void test1(char* s,char *st,...)
{
    char *ppt =(char*)&s;
    //printf("%p %p %p %p,",ppt,&s,&st,(char*)ppt+8);
    printf("%p %p %p %p\n", ppt, &s, &st, ppt + 8);
    printf(" %d\n",*(int*)(ppt + 4+4));//当是X64就加8 X86就加4因为内存对齐规则
    return;
}

int main()
{
    char *p = "Hello world";
    test1("111","eee",45234,23);
    //test(2, 2.2, "Hello world");x
    void *s = &p;
    printf("%s", *(char**)s);
    return 0;
}

其内涵主要在 va_list /start / arg/ end 的使用以及内存的理解。

此部分内容暂未完成,仍有很多需要理解的地方。

相关推荐
Tingjct10 小时前
【初阶数据结构-二叉树】
c语言·开发语言·数据结构·算法
飞机和胖和黄11 小时前
考研之王道C语言第三周
c语言·数据结构·考研
醉颜凉11 小时前
【LeetCode】打家劫舍III
c语言·算法·leetcode·树 深度优先搜索·动态规划 二叉树
一匹电信狗11 小时前
【LeetCode_21】合并两个有序链表
c语言·开发语言·数据结构·c++·算法·leetcode·stl
卢锡荣15 小时前
Type-c OTG数据与充电如何进行交互使用应用讲解
c语言·开发语言·计算机外设·电脑·音视频
v_for_van15 小时前
力扣刷题记录2(无算法背景,纯C语言)
c语言·算法·leetcode
二年级程序员15 小时前
动态内存管理
c语言
我能坚持多久15 小时前
D20—C语言文件操作详解:从基础到高级应用
c语言·开发语言
(❁´◡`❁)Jimmy(❁´◡`❁)16 小时前
CF2188 C. Restricted Sorting
c语言·开发语言·算法
想放学的刺客16 小时前
单片机嵌入式试题(第27期)设计可移植、可配置的外设驱动框架的关键要点
c语言·stm32·单片机·嵌入式硬件·物联网