c/c++的预处理定义 Stringizing Operator (#) Charizing Operator (#@) Token-Pasting Operator (##)
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='b';
注意:
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.连接后的实际参数名,必须为实际存在的参数名或是编译器已知的宏定义。