1.#的作用是,对这个变量替换后,再加双引号引起来,例如#define A(a) #a,那么有A(10) <--> "10"
对空格的处理
a. 忽略传入参数名前面和后面的空格。
如:str=example1( abc ); 将会被扩展成 str="abc";
b. 当传入参数名间存在空格时,编译器将会自动连接各个子字符串,用每个子字符串中只以一个空格连接,忽略其中多于一个的空格。
#define A1(name, type) type name_##type##_type
#define A2(name, type) type name##_##type##_type
则有:
A1(a1, int) <--> int name_int_type
A2(a1, int) <--> int a1_int_type
解释如下:
解释:
1) 在第一个宏定义中,"name"和第一个"_"之间,以及第2个"_"和第二个"type"之间没有被分隔,所以预处理器会把name_##type##_type解释成3段:
“name_”、“type”、以及“_type”,这中间只有“type”是在宏前面出现过
的,所以它可以被宏替换。
2) 而在第二个宏定义中,“name”和第一个“_”之间也被分隔了,所以
预处理器会把name##_##type##_type解释成4段:“name”、“_”、“type”
以及“_type”,这其间,就有两个可以被宏替换了。
3) A1和A2的定义也可以如下:
#define A1(name, type) type name_ ##type ##_type
<##前面随意加上一些空格>
#define A2(name, type) type name ##_ ##type ##_type
结果是## 会把前面的空格去掉完成强连接,得到和上面结果相同的宏定义
3.###可以看作是一个##加一个#,也就是先把后面的部分转化成字符串,再和前面的加在一起,例如#define A(name) "&"###name,则有 A(mike) <--> "&mike"
对空格的处理
a. 忽略传入参数名前面和后面的空格。
如:str=example1( abc ); 将会被扩展成 str="abc";
b. 当传入参数名间存在空格时,编译器将会自动连接各个子字符串,用每个子字符串中只以一个空格连接,忽略其中多于一个的空格。
如:str=exapme( abc def); 将会被扩展成 str="abc def";
#define A1(name, type) type name_##type##_type
#define A2(name, type) type name##_##type##_type
则有:
A1(a1, int) <--> int name_int_type
A2(a1, int) <--> int a1_int_type
解释如下:
解释:
1) 在第一个宏定义中,"name"和第一个"_"之间,以及第2个"_"和第二个"type"之间没有被分隔,所以预处理器会把name_##type##_type解释成3段:
“name_”、“type”、以及“_type”,这中间只有“type”是在宏前面出现过
的,所以它可以被宏替换。
2) 而在第二个宏定义中,“name”和第一个“_”之间也被分隔了,所以
预处理器会把name##_##type##_type解释成4段:“name”、“_”、“type”
以及“_type”,这其间,就有两个可以被宏替换了。
3) A1和A2的定义也可以如下:
#define A1(name, type) type name_ ##type ##_type
<##前面随意加上一些空格>
#define A2(name, type) type name ##_ ##type ##_type
结果是## 会把前面的空格去掉完成强连接,得到和上面结果相同的宏定义
3.###可以看作是一个##加一个#,也就是先把后面的部分转化成字符串,再和前面的加在一起,例如#define A(name) "&"###name,则有 A(mike) <--> "&mike"