可变参数

可变参数

 

说起可变参数,立即想到的就是printf(),scanf()了。可变参数就是有一个固定的格式控制参数,后面再根据格式控制参数接若干参数。

 

这里先讨论可变参数宏。

 

我们一般在Debug需要打印调试信息的时候,需要可变参数的宏。

 

一.vc(C99)的复杂宏

 

参考msdnhttp://msdn2.microsoft.com/en-us/library/ms177415.aspx)。

使用这种复杂宏时,省略号是格式控制参数,而标识符__VA_ARGS__用来插入另外的参数。__VA_ARGS__"..." 传递给宏。

 

例如:

#include <stdio.h>

 

#define CHECK1(x, ...) if (!(x)) {printf(__VA_ARGS__); }

#define CHECK2(x, ...) if ((x)) {printf(__VA_ARGS__); }

#define CHECK3(...) { printf(__VA_ARGS__);}

 

int main( )

{

        CHECK1(0, "here %s %s %s", "are", "some","varargs1(1)\n");

        CHECK1(1, "here %s %s %s", "are", "some","varargs1(2)\n");     // won'tprint

        CHECK2(0, "here %s %s %s", "are", "some","varargs2(3)\n");     // won'tprint

        CHECK2(1, "here %s %s %s", "are", "some","varargs2(4)\n");

        CHECK3("here %s %s %s", "are", "some","varargs3(5)\n");

        return 0;

}

 

输出为:

here are some varargs1(1)

here are some varargs2(4)

here are some varargs3(5)

 

. GCC中的复杂宏

 

GCC支持C99中的复杂宏,但G++不支持。GCC使用一种不同的语法,给可变参数一个名字,如同其它参数一样。

 

#define CHECK1(x,format, args...) if(x) {printf (format, args); }

显然这样易于描述而且可读性更强。

 

. CHECK"Some messges.")的情况

    这时由于宏展开后有个多余的逗号,,将导致编译错误.。为了解决这个问题,CPP使用一个特殊的“##”操作。

 

       #define CHECK(format, ...) printf ( format, ## __VA_ARGS__)

 

      这里,如果可变参数被忽略或为空,“##”操作将使预处理器去除掉它前面的那个逗号。如果在宏调用时,确实提供了一些可变参数,GNU CPP也会工作正常,它会把这些可变参数放到逗号的后面。

 

        四,类似printf()的技巧

 

        用一个被括弧括起来的“参数”来定义和调用宏,参数在宏扩展的时候成为类似printf()函数中的格式控制字符串那样的参数列表。

 

       #define CHECK(args) (printf("DEBUG: "), printf(args))

 

       用了括号。

 

 

/////////////////////////////////////////////////

 

  今天来说说宏。什么?宏也能可变参数?是的,你没有听错,带参数的宏和函数一样,同样支持可变参数。下面通过一个小程序加以说明。

 

#include

#include

 

#define OUTSCREEN(msg, ...)printf(msg,__VA_ARGS__)

 

int main(int argc, char* argv[])

{

    OUTSCREEN("Hello World!n%s", "__This is a MACRO!n");

    return 0;

}

 

     这个可变参数的宏是新的C99规范中新增的,目前似乎只有gcc支持(VC6.0的编译器不支持)。

 

#define OUTSCREEN(msg, ...) printf(msg,__VA_ARGS__)

int main(int argc, char* argv[])

{

    OUTSCREEN("Hello World!n%s", "__This is a MACRO!n");

    return 0;

}

    假如我们将上面的代码稍作一下修改,变成下面的样子。

 

#define OUTSCREEN(msg, ...) printf(msg,__VA_ARGS__)

 

int main(int argc, char* argv[])

{

    OUTSCREEN("Hello World!");

    return 0;

}

 

 

 

    注意我仅仅是将main函数里的OUTSCREEN做了修改,这时可变参数的个数为0了。但是编译的时候gcc却报错:

 

In function `main':

error: parse error before ')' token

 

    什么原因导致出错呢?把宏展开一下看看,原来是","惹得祸。那么这种参数个数可以为0的宏要怎么写呢?C99的规范没有定义这个,gcc对此做了扩展。重新定义OUTSCREEN宏如下:

 

#define OUTSCREEN(msg, ...) printf(msg,##__VA_ARGS__)

 

    当可变参数的个数为0时,这里的##起到把前面多余的","去掉,实际上变成了printf(msg),这样编译就能通过了。

    另外,__VA_ARGS__这个宏实在不利于记忆,gcc对此做了扩展,另一种可接受的定义方法为:

 

#define OUTSCREEN(msg, args...) printf(msg,##args)

.

###__VA_ARGS__ 

 

1.#

假如希望在字符串中包含宏参数,ANSI C允许这样作,在类函数宏的替换部分,#符号用作一个预处理运算符,它可以把语言符号转化程字符串。例如,如果x是一个宏参量,那么#x可以把参数名转化成相应的字符串。该过程称为字符串化(stringizing.

#incldue <stdio.h>

#define PSQR(x) printf("the squareof" #x "is %d.\n",(x)*(x))

int main(void)

{

   int y =4;

   PSQR(y);

   PSQR(2+4);

   return 0;

}

输出结果:

the square of y is 16.

the square of 2+4 is 36.

第一次调用宏时使用“y”代替#x;第二次调用时用“2+4"#x

2.##

##运算符可以用于类函数宏的替换部分。另外,##还可以用于类对象宏的替换部分。这个运算符把两个语言符号组合成单个语言符号。例如:

#define XNAME(n) x##n

这样宏调用:

XNAME(4)

展开后:

x4

程序:

#include <stdio.h>

#define XNAME(n) x##n

#define PXN(n) printf("x"#n"= %d\n",x##n)

int main(void)

{

   int XNAME(1)=12;//int x1=12;

   PXN(1);//printf("x1 = %d\n", x1);

   return 0;

}

输出结果:

x1=12

3.可变参数宏 ..._ _VA_ARGS__

__VA_ARGS__ 是一个可变参数的宏,很少人知道这个宏,这个可变参数的宏是新的C99规范中新增的,目前似乎只有gcc支持(VC6.0的编译器不支持)。

实现思想就是宏定义中参数列表的最后一个参数为省略号(也就是三个点)。这样预定义宏_ _VA_ARGS_ _就可以被用在替换部分中,替换省略号所代表的字符串。比如:

#define PR(...) printf(__VA_ARGS__)

int main()

{

   int wt=1,sp=2;

   PR("hello\n");

   PR("weight = %d, shipping = %d",wt,sp);

   return 0;

}

输出结果:

hello

weight = 1, shipping = 2

省略号只能代替最后面的宏参数。

#define W(x,...,y)错误!

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值