一、#
名称:字符串化操作符
其作用:将宏定义中的传入参数名转换成用一对双引号括起来的参数名字符串。
使用条件:只能用于有传入参数的宏定义中,且必须置于宏定义体中的参数名前
例1:
#define example1(instr) #instr
string str=example1(abc); 将会展成:string str="abc";
注意:对空格的处理
a、忽略传入参数名前面和后面的空格。
如:str=example1( abc ); 将会被扩展成 str="abc";
b、当传入参数名间存在空格时,编译器将会自动连接各个子字符串,用每个子字符串中只以一个空格连接,忽略其中多于一个的空格。
如:str=exapme1( abc def); 将会被扩展成 str="abc def";
例2:
// stringizer.cpp #include <stdio.h> #define stringer( x ) printf_s( #x "\n" ) int main() { stringer( In quotes in the printf function call ); stringer( "In quotes when printed to the screen" ); stringer( "This: \" prints an escaped double quote" ); }Output:
In quotes in the printf function call "In quotes when printed to the screen" "This: \" prints an escaped double quote"
二、##
名称:符号连接操作符
作用:将宏定义的多个形参名连接成一个实际参数名
使用条件:只能用于有传入参数的宏定义中
举例:
#define exampleNum(n) num##n
int num=exampleNum(9); 将会扩展成 int num=num9;
注意:
1、当用##连接形参时,##前后的空格可有可无。
如:#define exampleNum(n) num ## n 相当于 #define exampleNum(n) num##n
2、连接后的实际参数名,必须为实际存在的参数名或是编译器已知的宏定义
“##”是一种分隔连接方式,它的作用是先分隔,然后进行强制连接。
比如:
#define A1(name, type) type name_##type##_type
#define A2(name, type) type name##_##type##_type
1) 在第一个宏定义中,"name"和第一个"_"之间,以及第2个"_"和第二个"type"之间没有被分隔,所以预处理器会把name_##type##_type解释成3段:
“name_”、“type”、以及“_type”,这中间只有“type”是在宏前面出现过的,所以它可以被宏替换。
2) 而在第二个宏定义中,“name”和第一个“_”之间也被分隔了,所以预处理器会把name##_##type##_type解释成4段:“name”、“_”、“type”
以及“_type”,这其间,就有两个可以被宏替换了。
所以,
A1(a1, int); /* 等价于: int name_int_type; */
A2(a1, int); /* 等价于: int a1_int_type; */
三、@#
名称:字符化操作符
作用:将传入的单字符参数名转换成字符,以一对单引用括起来。
使用条件:只能用于有传入参数的宏定义中,且必须置于宏定义体中的参数名前。
举例:
#define makechar(x) @#X
a = makechar(b); 展开后变成了:a= 'b';
四、\
名称:行继续操作符
作用:当定义的宏不能用一行表达完整时,可以用"\"表示下一行继续此宏的定义。
注意:换行不能切断单词,只能在空格的地方进行。
转自:http://www.verydemo.com/demo_c441_i200593.html,对##的讲解有改正,
请看http://blog.csdn.net/jiangjingui2011/article/details/6706967。
see also http://www.kuqin.com/language/20080721/11906.html
MSDN上的相关文档 http://msdn.microsoft.com/en-us/library/wy090hkc.aspx