Chinaunix首页 | 论坛 | 博客
  • 博客访问: 1786287
  • 博文数量: 413
  • 博客积分: 8399
  • 博客等级: 中将
  • 技术积分: 4325
  • 用 户 组: 普通用户
  • 注册时间: 2011-06-09 10:44
文章分类

全部博文(413)

文章存档

2015年(1)

2014年(18)

2013年(39)

2012年(163)

2011年(192)

分类: C/C++

2011-11-07 10:42:24

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


#incldue

#define PSQR(x) printf("the square of" #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
#define XNAME(x) x##4
#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;
}


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


#define PR(...) printf(_ _VA_ARGS_ _)

PR("hello");-->printf("hello");
PR("weight = %d, shipping = $.2f",wt,sp);
    -->printf("weight = %d, shipping = $.2f",wt,sp);


省略号只能代替最后面的宏参数
#define W(x,...,y)错误!

// variadic.c -- variadic macros

#include

#include


#define PR(X, ...) printf("Message" #X ": " _ _VA_ARGS_ _)

int main(void)

{

    double x = 48;

    double y;

    y = sqrt(x);

    PR(1, "x = %g\n", x);

    PR(2, "x = %.2f, y = %.4f\n", x, y);

    return 0;

}

In the first macro call, X has the value 1, so #X becomes "1". That makes the expansion look like this:

(#为参数加双引号。)


print("Message " "1" ": " "x = %g\n", x);

Then the four strings are concatenated, reducing the call to this:

print("Message 1: x = %g\n", x);

Here's the output:

Message 1: x = 48

Message 2: x = 48.00, y = 6.9282

Don't forget, the ellipses have to be the last macro argument:

#define WRONG(X, ..., Y) #X #_ _VA_ARGS_ _ #y(这个是错误的例子。)

补充:

利用##和__VA_ARGS__来调试代码:

#define  __DEBUG__
#ifdef     __DEBUG__
#define  DEBUG(format, ...)      fprintf(stderr, format, ##__VA_ARGS__)
#define  INFO(format, ...)          printf(format, ##__VA_ARGS__)
#else
#define  DEBUG(format, ...) 
#endif

注意:
不能定义成 #define DEBUG2(format, ...)    printf(format, __VA_ARGS__)
因为这样的话调用DEBUG2("debug info:"); 
会被扩展成printf("debug info:", );注意这里最后有一个逗号 "," !显然不合语法。

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



#define gf_log(dom, levl, fmt...) do {                                  \                                                                                     

                if (levl <= gf_log_loglevel)                            \

                        _gf_log (dom, __FILE__, __FUNCTION__, __LINE__, \

                                 levl, ##fmt);                          \

} while (0)

  

阅读(943) | 评论(0) | 转发(0) |
给主人留下些什么吧!~~