C语言中的#,##

来源:互联网 发布:crm软件哪个好 编辑:程序博客网 时间:2024/05/22 14:48

C语言中的#,##

今天在分析一份源码的时候遇到了如下代码:

#define UF_CALL(X)  (report(__FILE__,__LINE__,#X,(X)))

static int report (char* file, int line,char * call,int irc)

{

if(irc)

{

char msg[133];

printf("%s,line%d:%s\n",file,line,call);

(UF_get_fail_message(irc,msg))?printf("returned a %d\n",irc):printf("returned error %d:%s\n",irc,msg);

}

return (irc);

}

     网上参考:

[cpp] view plaincopyprint?
  1. #define CASE(x,y)  case x: _bsd_setlasterror(anApplInstance, BSD_ ## y); break  

觉得其中的##很有意思,于是就将这个知识点整理一下。#和##都是预处理指令,我们先看#。

#用来把参数转换成字符串,请看下面的两个例子。

注:#x  要在 #define中使用,否则会报错。

例子一:让标示符原样输出

[cpp] view plaincopyprint?
  1. #include <stdio.h>  
  2. #include <iostream>  
  3.   
  4. #define P(A) printf("%s:%d\n",#A,A);  
  5.   
  6. int main(int argc, char **argv)  
  7. {  
  8.     int a = 1, b = 2;  
  9.     P(a);  
  10.     P(b);  
  11.     P(a+b);  
  12.     system("pause");  
  13. }  

例子二:也是原样输出,只是此处这就给的是一个整形数,达到和上面不一样的效果。
[cpp] view plaincopyprint?
  1. #define SQR(x) printf("The square of x is %d.\n", ((x)*(x)));  
如果这样使用宏:SQR(8);
则输出为:The square of x is 64.
注意到没有,引号中的字符x被当作普通文本来处理,而不是被当作一个可以被替换的语言符号。
假如你确实希望在字符串中包含宏参数,那我们就可以使用“#”,它可以把语言符号转化为字符串。上面的例子改一改:
[cpp] view plaincopyprint?
  1. #define SQR(x) printf("The square of "#x" is %d.\n", ((x)*(x)));  
再使用:SQR(8);
则输出的是:The square of 8 is 64.
和#运算符一样,##运算符可以用于宏函数的替换部分。这个运算符把两个语言符号组合成单个语言符号。看例子:
[cpp] view plaincopyprint?
  1. #define XNAME(n) x ## n  
如果这样使用宏:XNAME(8)
则会被展开成这样:x8
##就是个粘合剂,将前后两部分粘合起来,也就是有“字符化”的意思。但是“##”不能随意粘合任意字符,必须是合法的C语言标示符。在单一的宏定义中,最多可以出现一次“#”或“##”预处理操作符。如果没有指定与“#”或“##”预处理操作符相关的计算次序,则会产生问题。为避免该问题,在单一的宏定义中只能使用其中一种操作符(即,一份“#”或一个“##”,或都不用)。除非非常有必要,否则尽量不要使用“#”和“##”。
原创粉丝点击