在代码中,看到了包含##的宏。
例如:#define clear_opt(o, opt) o &= ~EXT2_MOUNT_##opt
于是,一起看了下# 和 ##在宏中的用法。
wiki中的解释如下:
# 为 字符串转换符, 将包含的字段转换为字符串。
## 为 字段连接符,将两个字段 连接为 一个字段。
wiki的示例有一段示例比较有意思:
#define xstr(s) str(s)
#define str(s) #s
#define foo 4
str
(
foo
)
// outputs "foo"
xstr
(
foo
)
// outputs "4"
给出的解释是:无法将宏中附带的文本或者字符串 合并到宏中。而应该使用字符串常量 和 字符串参数。C编译器会将所有的字符串常量合并为一个长的字符串。
使用如下代码验证:
19 #include <stdio.h>
20
21 #define foo 4
22 #define STR(s) #s
23 #define XSTR(s) STR(s)
24 #define CONS(a,b) STR(a##o##b)
25 #define XCONS(a,b) XSTR(a##o##b)
26
27 int main() {
28 printf(STR(foo));
29 printf("\n");
30 printf(XSTR(foo));
31 printf("\n");
32 printf("%s\n", CONS(f,o));
33 printf("%s\n", XCONS(f,o));
34 return 0;
35 }
得到结果:
:!./a.out
foo
4
foo
4
Press ENTER or type command to continue
有些宏的用法,确实和正常的思维存在差异。小心。
=====================华丽的分割线=================
回想这个宏的例子,显露出了宏的一个缺陷。
宏所定义的字段是没有数据类型的。这个字段不一定会以普通程序的逻辑来运行。