• C和C++中的不定参数


    在初学C的时候,我们都会用到printf函数来写Hello World的程序.在我们看printf函数的声明时,会看到类似于下面代码

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

    另外,在我们学习C和C++的时候,函数的声明总是确定个数和类型的,而我们在用printf的时候,却可以一次输出多个参数.

    这就是我们要提的不定参数了.

    在32位的C和C++编程中,函数调用是有规约的,并且各个编译器也基本达成了一致,尽管他们编译出的东西基本不能通用.关于调用规约的东西,可以参考一下维基百科.

    在64位的C和C++编程中,就没有调用规约的概念了,基本上做到了统一,但是不同的编译器的传参方式却不尽相同,这里我们不去讨论了,有兴趣的可以编译到汇编代码查看.

    不定参数的函数调用方式为cdec方式的,也就是由调用者来恢复参数栈,这个不难理解,因为被调用的函数无从得知有多少个参数传进来,所以不可能知道如何恢复栈.

    如果上面写的你看不懂,不要紧,你可以用google搜一下,相信很快你就会明白了,不搜也不要紧.

    在使用不定参数时,我们会用到三个宏,分别是

    va_start , va_arg 和 va_end

    还有一个类型 va_list

    它们都定义在 stdarg.h 或者 cstdarg (C++)里,使用时记得引入.

    其中,va_start用来用于初始化va_list, va_arg用来读取va_list中的参数,当所有的读取都结束后要用va_end来释放va_list.

    下面写一个示例

    复制代码
    #include<stdlib.h>
    #include<stdio.h>
    #include<stdarg.h>
    int sum(int count, ...);
    int main()
    {
        int nS = sum(3,1,2,3);
        
        printf("%d
    ", nS);
        return 0;
    }
    int sum(int count, ...)
    {
        int _sum,arg,i;
        va_list arg_ptr;
        _sum = 0;
        va_start(arg_ptr,count);
        for(i=0; i < count; ++i)
        {
            arg = va_arg(arg_ptr,int);
            _sum += arg;
        }
        va_end(arg_ptr);
        return _sum;
    }
    复制代码

    上面的例子比较简单,后续参数个数由第一个参数指定,而且类型默认都是int类型的.

    在printf函数中,后续参数个数是由第一个格式字符串来指定的,并且指定了参数类型,比如%d 说明对应的参数是整形而%f 对应的是浮点类型.

    接下来我们看看这几个宏.

    va_start是用来初始化va_list的,第一个参数是参数表的指针,第二个参数是不定参数前的最后一个参数.

    va_arg 是用来读取不定参数,第一个参数是参数表的指针,第二个参数是参数的类型.函数本身并不知道参数的类型,所以使用不当会导致出错.

    va_end是用来释放va_list占用的资源的,只有一个参数,就是要释放的va_list.

    最后,我们通过printf函数来总结一下使用不定参数的一些规范:

    1. 函数本身必须有办法知道不定参数的类型,比如printf通过格式化字符串通知函数后续参数的每个参数的类型,其中的%d类形的格式与后续的参数是一一对应的.在我写的示例代码中,是默认约定了所有参数都是int 类型的.

    2. 函数必须能知道参数结束的地方,printf函数是由格式输出字符串来知道的,当没有类似%d或%f 这种字符出现时,参数就结束了.我给出的例子中,是通过第一个参数给定了后续参数的个数的.

    3.调用必须严格按照调用的约定来做,而且不定的参数是不会自动转型的,比如当我们 printf("%d",3.3) 会发现输出的不是3,就是因为3.3作为一个浮点数传入,而不会因为格式字符串中的%d自动转成整数.要想得到预期的结果,需要写成下面这样 printf("%d",(int)3.3)

  • 相关阅读:
    自定义指令directive
    angular中的表单验证
    ng-init,ng-controller,ng-model
    Redis执行lua脚本,key不存在的返回值
    消息队列对比
    数据库设计范式
    网络IO模型
    .NET 线程、线程池
    异步和多线程
    Memcache知识点
  • 原文地址:https://www.cnblogs.com/fnlingnzb-learner/p/6401833.html
Copyright © 2020-2023  润新知