C++的特殊预处理定义#、##和#@

c/c++的预处理定义:

一。Stringizing Operator (#)
在c和c++中数字标志符#被赋予了新的意义,即字符串化操作符。其作用是:将宏定义中的传入参数名转换成用一对双引号括起来的参数名字符串。其只能用于有传入参数的宏定义中,且必须置于宏定义体中的参数名前。
如:
 #define example(instr) printf("the input string is:/t%s/n",#instr)
 #define example1(instr) #instr
当使用该宏定义时:
example(abc); 在编译时将会展开成:printf("the input string is:/t%s/n","abc");
string str=example1(abc); 将会展成:string str="abc";

注意:
1。对空格的处理
a。忽略传入参数名前面和后面的空格。
如:str=example1(   abc  ); 将会被扩展成  str="abc";
b.当传入参数名间存在空格时,编译器将会自动连接各个子字符串,用每个子字符串中只以一个空格连接,忽略其中多余一个的空格。
如:str=exapme( abc    def); 将会被扩展成  str="abc def";

2。转义字符
a. 某些形式的传入参数名中,若存在特殊字符,编译器会自动为其添加转义字符号'/'。
如:
string str=example1("escap'/e"); 相当于:str="/"escap/'//e/"";
b. vc6.0和vc7.0并不能正确的解析所有需要特殊字符的情况。此时会给出错误报告:error C2001: 常数中有换行符。
如:
example1(abc/'); //此处报警 error C2001: 常数中有换行符

二。Charizing Operator (#@)
字符化操作符。其同样只能用于有传入参数的宏定义中,且必须置于宏定义体中的参数名前。作用,将传的单字符参数名转换成字符,以一对单引用括起来。
如:
#define exampleChar(inchar) #@inchar
使用该宏定义:
char a=exampleChar(a); 将会被扩展成:char a='a';

注意:
vc6.0和vc7.0中默认的类型转换中,可以将int截断成char。因此,参数名中最多不能超过4个字符。
如:
char a=example(abcd) 将会截断成  a='d'。同时编译器会给出:warning C4305: “=” : 从“int”到“char”截断

三。Token-Pasting Operator (##)
##:符号连接操作符。
宏定义中:参数名,即为形参,如#define sum(a,b) (a+b);中a和b均为某一参数的代表符号,即形式参数。
而##的作用则是将宏定义的多个形参成一个实际参数名。
如:
#define exampleNum(n) num##n
int num9=9;
使用:
int num=exampleNum(9); 将会扩展成 int num=num9;

注意:
1.当用##连接形参时,##前后的空格可有可无。
如:#define exampleNum(n) num ## n 相当于 #define exampleNum(n) num##n
2.连接后的实际参数名,必须为实际存在的参数名或是编译器已知的宏定义。 

原文地址:https://www.cnblogs.com/enumhack/p/4d6bae9e5cfea8d2366e8455b3d8e42d.html