C语言 深度探究具有不定参数的函数

简介: C语言 深度探究具有不定参数的函数

C语言 深度探究具有不定参数的函数


✨博主介绍

前言

C语言 stdarg.h

示例

ta的原理

函数传参数的本质

_INTSIZEOF(n)

其他宏

练习 实现printf

💫点击直接资料领取💫


✨博主介绍

💂 个人主页:苏州程序大白


💂 个人社区:CSDN全国各地程序猿


🤟作者介绍:中国DBA联盟(ACDU)成员,CSDN全国各地程序猿(媛)聚集地管理员。目前从事工业自动化软件开发工作。擅长C#、Java、机器视觉、底层算法等语言。2019年成立柒月软件工作室,2021年注册苏州凯捷智能科技有限公司


💅 有任何问题欢迎私信,看到会及时回复


👤 微信号:stbsl6,微信公众号:苏州程序大白


💬如果文章对你有帮助,欢迎关注、点赞、收藏(一键三连)


🎯 想加入技术交流群的可以加我好友,群里会分享学习资料


前言


C语言虽然不支持泛型编程(至少C98是这样的),但是C语言却支持不定参数的函数,这里我深究一下里面的原理,并且学会它的使用,自己简单实现一个简单的printf函数。 注:这里使用的IDE为 vs2022


至于如何实现不定参数的函数呢?这里可以看一下标准库是如何定义的:


    _Check_return_opt_
    _CRT_STDIO_INLINE int __CRTDECL printf(
        _In_z_ _Printf_format_string_ char const* const _Format,
        ...)


这里char const* const _Format ,显然就是我们传入格式字符串,后面 出现了 …,这个类型没见过,那它应该是实现可变参数的关键了。在C语言中… 三点就表示不定参数,这里我们又遇到了一个新的问题,传入了不定参数之后如何拿出不定参数?使用几个宏就完成这一个操作,没错就是宏。


C语言 stdarg.h


定义


在了解 ta的原理之前,还是学会如何使用ta。


宏 描述

void va_start(va_list ap, last_arg) 这个宏初始化 ap 变量,使ap指向起始的参数(last_arg)

type va_arg(va_list ap, type) 获取下一个类型为type的参数

void va_end(va_list ap) 释放ap

va_copy(destination, source) 拷贝ap的内容

提示:va_start va_arg va_end 务必按照此顺序使用


示例


光看这个表格实在是难以理解 还是实操一个吧。


#include<stdio.h>
#include<stdarg.h>
//实现 args_nums个int类型的数相加
int sum(int args_num, ...)
{
    va_list ap;
    int sum = 0;
    va_start(ap, args_num);//1. 初始化 ap 
    for (; args_num > 0; --args_num)
        sum += va_arg(ap, int);//2. 获取 下一个int类型的参数
    va_end(ap);//结束使用
    return sum;
}
int main()
{
    printf("sum: %d", sum(5, 1, 2, 3, 4, 5));
    return 0;
}


仔细阅读该程序,我们可以大致了解ta的基本使用


1、va_start(ap, args_num) 初始化ap。


2、va_arg(ap, int) 获取下一个int类型参数。


3、va_end(ap) 结束使用。


4、补充:stdarg.h 并没有提供帮我们判断不定参数有多个的方法,这里我是用 传入一个args_num来标记有多个不定参数,不要以为我们必须传入一个int来标记,我们可以采取其他方法的(后面补充)。


5、这里我们必须传入一个确定的参数作为第一个参数,因为 va_start 需要一个确定的参数初始化。


运行结果:


ea1764463deb4db0824b20e048d8ffcb.png


ta的原理


函数传参数的本质


C语言是最接近汇编的一门语言,函数传参的本质到底是什么,简单一句话 ——将参数压栈,如何你有汇编的经历的话,就知道如果要给一个过程传入参数就需要你提前将传入的参数压入栈中,C语言就是这样做的,当然控制压栈这么麻烦的操作编译器在编译的过程中就帮你完成了。当然这要拿出汇编中的一个知识点,每次压栈和出栈的基本单位不是字节,而是当前CPU的字长为单位的,比如 32位那么每次压栈就是以4字节位基本单位的。


现在我们研究一下,多个参数的压栈顺序,是从左到右还是反之?


#define VNAME(val) (#val) // 获取val变量的名字
typedef struct test { char c[6]; } test; //定义结构体 test
void foo(test a, char b,int c )
{
    printf("%s addr: %p\n",VNAME(a),&a);
    printf("%s addr: %p\n", VNAME(b), &b);
    printf("%s addr: %p\n", VNAME(c), &c);
}
int main()
{
    test t = { "123456" };
    foo(t, 2, 3);
    return 0;
}


补充:C程序栈底为高地址,栈顶为低地址。


在X86的环境下,我们在第8行打入断点,使用内存查看工具查看内存。


38013433f8ef4b1eafcf552741c864f4.png


我们发现大小只有 1字节的b都占了4字节的大小,大小为6的a占了8字节,这一点是万确适应 前面所说的 32位那么每次压栈就是以4字节位基本单位的,如果是64为的话,那么char一定会占8字节。


输出:


adc04689d48e4318beb7eefb25a64f78.png


我们发现下 从 c到 a地址越来越小,说明c先入栈,后面才进b和a, 得出结论 C语言函数参数入栈顺序为从右至左。


如果我们得到了第一个参数的地址,那么我们可以根据参数的所占空间来确定下一个参数的地址,那么我们不就是获取了下一个参数的值了吗?C语言也是这样想的。例如:知道 a的地址为 010FFAA4 ,A所占空间为8,那么b的地址一定为 &a+8。


我们简单验证一下:更具 a的地址获取 b和c的值。


    char* p = &a;
    char bb = *(char*)(p += 8);
    int cc = *(int*)(p += 4);
    printf("b : %d,c : %d\n", bb, c);


这里强调一点的是,咱们使用 p 为char*,原因很简单,如果是其他类型指针如 int ,那么 p+8 却偏移了48=32字节位置,而不是 8字节。那么我有理由相信 va_list就是 char


我们这个唯一的缺点就是只解决这一个函数的特例,无法自定义,如果有函数可以帮我们求出 偏移量就好了。


这里我们就解开庐山真面目,看看标准库是如何定义他们的。(这里经过了多次跳转)。


    typedef char* va_list;   //和我想得一样
  #define _INTSIZEOF(n)          ((sizeof(n) + sizeof(int) - 1) & ~(sizeof(int) - 1)) //求n类型 在栈中所占空间
    #define __crt_va_start_a(ap, v) ((void)(ap = (va_list)_ADDRESSOF(v) + _INTSIZEOF(v))) //初始化 ap
    #define __crt_va_arg(ap, t)     (*(t*)((ap += _INTSIZEOF(t)) - _INTSIZEOF(t)))//获取下一个变量
    #define __crt_va_end(ap)        ((void)(ap = (va_list)0))//结束使用ap
  #define va_copy(destination, source) ((destination) = (source))//拷贝


这里 我们单独解释一下每个宏原理。


_INTSIZEOF(n)


#define _INTSIZEOF(n)          ((sizeof(n) + sizeof(int) - 1) & ~(sizeof(int) - 1))


_INTSIZEOF(n)整个做的事情就是将n的长度化为int长度的整数倍,比如n为5,二进制就是101b,int长度为4,二进制为100b,那么n化为int长度的整数倍就应该为8。换一句话说它就帮我们求出变量间的偏移量。


~是位取反的意思。~(sizeof(int) – 1) )就应该为~(4-1)=~(00000011b)=11111100b,这样任何数& ~(sizeof(int) – 1) )后最后两位肯定为0,就肯定是4的整数倍了。(sizeof(n) + sizeof(int) – 1)就是将大于4m但小于等于4(m+1)的数提高到大于等于4(m+1)但小于4(m+2)(m为0,1,2,…),这样再& ~(sizeof(int) – 1) )后就正好将原长度补齐到4的倍数了。


MSVC是这样实现的,我们可以看看 GNUC是如何实现的:


#define __va_rounded_size(TYPE)  \  //名字虽然不一样但是功能是一样的,毕竟这是不同的厂家
  (((sizeof (TYPE) + sizeof (int) - 1) / sizeof (int)) * sizeof (int)) 


显然这个更容易理解一些,(sizeof (TYPE) + sizeof (int) - 1)/ sizeof (int)求得type所占空间至少为 int类型的多少整数倍,再乘以4刚好实现了 长度补齐到4的倍数。比如__va_rounded_size(char)= ((1+3)/4)4= 4。


个人感觉 MSVC的效率更好一点,毕竟是 逻辑运算,当然GNUC的方法更加容易理解,我研究微软的实现方式还是花了不少的时间的。


其他宏


剩下三个宏,非常容易理解,我就不在解释了,大伙可以对照实验和定义,很快你就会明白其中的奥义。


练习 实现printf


这里我们仅仅只是练习多个参数的传递 而不是深入底层地实现printf这个函数,如果你愿意可以去看看 vprintf的实现,有非常多的奥秘值得探究。


这里放出我的代码,非常小儿科。


#include<stdio.h>
#include<stdarg.h>
int myPrintf(const char* string, ...);
int main()
{
    //printf("sum: %d", sum(5, 1, 2, 3, 4, 5));
    myPrintf("helloworld%d%s%c\na", 15, "122", 'z');
    return 0;
}
//大致思路 将string 中的 %d %c \n 等 替换成 对应的字符 存储再 buffer中
//这里只实现了 %d %c %s \n
int myPrintf(const char* string, ...)//返回实际输出字符个数
{
    char buffer[BUFSIZ];
    int temp = 0;
    va_list arg;
    char* p_string = NULL;
    char* p_buffer = buffer;
    char* p_temp = NULL;
    int counter = 0;//输出字符长度
    int number = 0;
    int foo = 0;
    va_start(arg, string);
    for (counter = 0, p_string = string; *(p_string) != '\0';)
    {
        switch (*p_string)
        {
        //处理 %
        case '%':
        {
            switch (*++p_string)
            {
            case 'd':
                temp = va_arg(arg, int);
                foo = temp;
                //获取 位数
                while (foo)
                {
                    number++;
                    counter++;
                    foo /= 10;
                }
                foo = temp;
                //整数转字符串
                while (number)
                {
                    *(p_buffer + number - 1) = (foo % 10)+'0';
                    foo /= 10;
                    number--;
                }
                p_buffer += number;
                break;
            case 'c':
                temp = va_arg(arg, int);
                *(p_buffer++) = temp;
                break;
            case 's':
                p_temp = va_arg(arg, char*);
                while (*p_temp !='\0')
                {
                    *(p_buffer++) = *(p_temp++);
                    counter++;
                }
                break;
            default:
                break;
            }
            ++p_string;
            break;
        }
        //处理 转移字符
        case '\\':
        {
            p_string++;
            switch (*p_string)
            {
            case 'n':
                *(p_string++) = '\n';
                break;
            default:
                break;
            }
            break;
        }
        default:
            *(p_buffer++) = *(p_string++);
            counter++;
        }
    }
    *(p_buffer++) = '\0';
    va_end(arg);
    p_buffer = NULL;
    puts(buffer);
    return counter;
}


相关文章
|
25天前
|
存储 C语言 开发者
【C语言】字符串操作函数详解
这些字符串操作函数在C语言中提供了强大的功能,帮助开发者有效地处理字符串数据。通过对每个函数的详细讲解、示例代码和表格说明,可以更好地理解如何使用这些函数进行各种字符串操作。如果在实际编程中遇到特定的字符串处理需求,可以参考这些函数和示例,灵活运用。
51 10
|
25天前
|
存储 程序员 C语言
【C语言】文件操作函数详解
C语言提供了一组标准库函数来处理文件操作,这些函数定义在 `<stdio.h>` 头文件中。文件操作包括文件的打开、读写、关闭以及文件属性的查询等。以下是常用文件操作函数的详细讲解,包括函数原型、参数说明、返回值说明、示例代码和表格汇总。
43 9
|
25天前
|
存储 Unix Serverless
【C语言】常用函数汇总表
本文总结了C语言中常用的函数,涵盖输入/输出、字符串操作、内存管理、数学运算、时间处理、文件操作及布尔类型等多个方面。每类函数均以表格形式列出其功能和使用示例,便于快速查阅和学习。通过综合示例代码,展示了这些函数的实际应用,帮助读者更好地理解和掌握C语言的基本功能和标准库函数的使用方法。感谢阅读,希望对你有所帮助!
34 8
|
25天前
|
C语言 开发者
【C语言】数学函数详解
在C语言中,数学函数是由标准库 `math.h` 提供的。使用这些函数时,需要包含 `#include <math.h>` 头文件。以下是一些常用的数学函数的详细讲解,包括函数原型、参数说明、返回值说明以及示例代码和表格汇总。
43 6
|
25天前
|
存储 C语言
【C语言】输入/输出函数详解
在C语言中,输入/输出操作是通过标准库函数来实现的。这些函数分为两类:标准输入输出函数和文件输入输出函数。
178 6
|
25天前
|
存储 缓存 算法
【C语言】内存管理函数详细讲解
在C语言编程中,内存管理是至关重要的。动态内存分配函数允许程序在运行时请求和释放内存,这对于处理不确定大小的数据结构至关重要。以下是C语言内存管理函数的详细讲解,包括每个函数的功能、标准格式、示例代码、代码解释及其输出。
55 6
|
25天前
|
C语言 开发者
【C语言】断言函数 -《深入解析C语言调试利器 !》
断言(assert)是一种调试工具,用于在程序运行时检查某些条件是否成立。如果条件不成立,断言会触发错误,并通常会终止程序的执行。断言有助于在开发和测试阶段捕捉逻辑错误。
35 5
|
1月前
|
存储 人工智能 算法
数据结构实验之C 语言的函数数组指针结构体知识
本实验旨在复习C语言中的函数、数组、指针、结构体与共用体等核心概念,并通过具体编程任务加深理解。任务包括输出100以内所有素数、逆序排列一维数组、查找二维数组中的鞍点、利用指针输出二维数组元素,以及使用结构体和共用体处理教师与学生信息。每个任务不仅强化了基本语法的应用,还涉及到了算法逻辑的设计与优化。实验结果显示,学生能够有效掌握并运用这些知识完成指定任务。
60 4
|
1月前
|
C语言
c语言调用的函数的声明
被调用的函数的声明: 一个函数调用另一个函数需具备的条件: 首先被调用的函数必须是已经存在的函数,即头文件中存在或已经定义过; 如果使用库函数,一般应该在本文件开头用#include命令将调用有关库函数时在所需要用到的信息“包含”到本文件中。.h文件是头文件所用的后缀。 如果使用用户自己定义的函数,而且该函数与使用它的函数在同一个文件中,一般还应该在主调函数中对被调用的函数做声明。 如果被调用的函数定义出现在主调函数之前可以不必声明。 如果已在所有函数定义之前,在函数的外部已做了函数声明,则在各个主调函数中不必多所调用的函数在做声明
38 6
|
1月前
|
存储 算法 程序员
C语言:库函数
C语言的库函数是预定义的函数,用于执行常见的编程任务,如输入输出、字符串处理、数学运算等。使用库函数可以简化编程工作,提高开发效率。C标准库提供了丰富的函数,满足各种需求。