C语言怎样写参数个数可变的宏?

一、问题

在C语⾔中存在参数个数可变的函数,那么是否也存在参数个数可变的宏呢?如果存在,怎样写参数个数可变的宏呢?

二、解答

在C语⾔中存在参数个数可变的宏,⾸先⼤致了解⼀下什么是参数个数可变的函数, printf( )和 scanf( )函数是使⽤最频繁的参数个数可变的函数。

下⾯这个例⼦是参数个数可变的函数的应⽤,相应代码如下。

cpp 复制代码
#include <stdio.h>
#include <stdarg.h>
int func(int first, int second, ...)
{
    int s = 0, t = first;
    va_list v1;
    va_start(v1, first);
    while (t != -1)
    {
        s += t;
        t = va_arg(v1, int);
    }
    va_end(v1);
    return s;
}
int main(int argc, char *argv[])
{
    printf("个数不定的参数的和为s=%d\n", func(50, 10, 12, 20, -1));
    printf("个数不定的参数的和为s=%d\n", func(50, 10, 12, 20, 8, -1));
    return 0;
}

此例实现了⼀个参数个数不确定的求和的函数 func( ),在 func( )函数中输⼊任意个数的参数,输出这⼏个参数的和。例如,输⼊参数 50、10、 12、 20、-1,可以得到这⼏个参数的和为92。其中,-1是参数结束标志。当输⼊ 50、10、12、20、8、-1时,函数得到的计算结果为100。

此程序的运⾏结果如下所示。

cpp 复制代码
个数不定的参数的和为s=92
个数不定的参数的和为s=100

上述例⼦中,函数中⼀开始定义了⼀个 v_list 型变量 v,该变量⽤来访问可变参数, 实际上就是指针,接着使⽤ va_start 使v1指向第⼀个参数,然后再使⽤ va_arg 来遍历每⼀ 个参数,va_arg 返回参数列表中的当前参数并使 v1指向参数列表中的下⼀个参数。最后通 过 va_end 把v1指针清为 NULL。在这⾥,va_ start,va_arg,va_end 其实都是宏。

参数个数可变的宏与参数个数可变的函数⼤体是相似的,都是参数个数不确定。了解了如何编写⼀个参数个数可变的函数,那么如何编写⼀个参数个数可变的宏也就可以理解了。

一种流⾏的技巧是⽤⼀个单独的⽤括号括起来的"参数"定义和调⽤宏,参数在宏扩展的时候成为类似 printf( )那样的函数的整个参数列表。例如:

cpp 复制代码
#define DEBUG(args) (printtf("DEBUG:"), printf args)
if (n != 0)
    DEBUG(("n is %d\n", n));

注意:

调⽤者必须记住使⽤⼀对额外的括号括起,如"DEBUG(("n is %d\n",n));"。

GCC 有⼀个扩展可以让函数式的宏接受可变个数的参数。但这并不是标准,另⼀种可能的解决⽅案是根据参数个数使⽤多个宏,如 DEBUG1、 DEBUG2 等,或者⽤逗号表示成如下形式。

cpp 复制代码
#define DEBUG(args) (printf("DEBUG:"),printf args)
#define_,
DEBUG("i=%d"_i);

C99 引⼊了对参数个数可变的函数式宏的正式⽀持。在宏"原型" 的末尾加上符号"..."宏定义中的伪宏_VA_ARGS_就会在调⽤时替换成可变参数。例如:

cpp 复制代码
#define dgbmsg(fmt...)\
printf(fmt,_VA_ARGS_)

这⾥"..."表示可变参数。

三、总结

起初,可变参数还只是应⽤在真正的函数中,不能应⽤在宏中。直到 C99 编译器标准的出现,改变了这种局⾯,它允许定义参数可变的宏。虽然 GCC 也可以定义参数个数可变的宏,但是它并不是标准的。

相关推荐
R-G-B13 分钟前
哈希表(hashtable),哈希理论,数组实现哈希结构 (C语言),散列理论 (拉链发、链接发),散列实现哈希结构,c++ 实现哈希
c语言·哈希算法·散列表·哈希表·数组实现哈希结构·散列实现哈希结构·c++ 实现哈希
历程里程碑17 分钟前
C++ 6 :string类:高效处理字符串的秘密
c语言·开发语言·数据结构·c++·笔记·算法·排序算法
蓝色汪洋18 分钟前
luogu迷宫寻路
算法
武帝为此20 分钟前
【字典树 C++ 实现】
开发语言·c++
悟能不能悟24 分钟前
java 设置日期返回格式的几种方式
java·开发语言
未来之窗软件服务24 分钟前
幽冥大陆(四十八)P50酒店门锁SDK 苹果object c语言仙盟插件——东方仙盟筑基期
c语言·开发语言·酒店门锁·仙盟创梦ide·东方仙盟·东方仙盟sdk
while(1){yan}1 小时前
基于IO流的三个小程序
java·开发语言·青少年编程
木头左1 小时前
自适应门限动态调整算法在量化交易策略中的应用
算法
deepdata_cn1 小时前
非线性规划(NLP)算法
算法
王燕龙(大卫)1 小时前
rust:线程
开发语言·rust