# 的功能
# 的功能是将其后面的宏参数进行字符串化操作(stringfication),简单说就是在对它所引用的宏变量通过替换后在其左右各加上一个双引号。
1. #define WARN_IF(exp)
2. do {
3. if (exp)
4. fprintf(stderr, "Warning: " #exp "n");
5. } while(0)
现在在程序中以下面的方式调用这个宏:
1. WARN_IF (divider == 0);
那么在编译时,上面的这句话被扩展为:
1. do { if (divider == 0) fprintf(stderr, "Warning: " "divider == 0" "n"); } while(0);
这样每次divider(除数)为0的时候便会在标准错误流上输出一个提示信息。
## 的功能
## 称为连接符号(concatenator),由两个# 号组成,其功能是在带参数的宏定义中将两个子串(token)联接起来,从而形成一个新的子串。但它不可以是个或者一个子串。所谓的子串(token)就是指编译器能够识别的小语法单元,注意这里连接的对象是token就行,而不一定是宏的变量。
下面举个例子来看看它们是怎样工作的。假设程序中已经定义了这样一个带参数的宏:
1. #define LINK_MULTIPLE(a,b,c,d) a##_##b##_##c##_##d
现在在程序中以下面的方式调用这个宏:
1. struct _record_type LINK_MULTIPLE(name,company,position,salary);
那么在编译时,上面的这句话被扩展为:
1. struct _record_type name_company_position_salary;
综合举例
举一个综合# 和## 的例子。假设程序中已经定义了这样一个带参数的宏:
1. #define paster( n ) printf( "token" #n " = %d", token##n )
现在在程序中以下面的方式调用这个宏:
1. int token9 = 9;
2. paster( 9 );
那么在编译时,上面的paster( 9 ); 这句话被扩展为:
1. printf( "token" "9" " = %d", token9 );
注意到在这个例子中, paster(9); 中的这个9 被原封不动的当成了一个字符串,与”token”连接在了一起,从而成为了token9 。而#n 也被”9”所替代。 可想而知,上面程序运行的结果就是在屏幕上打印出
1. token9=9
理解了这些嵌入式Linux的特殊符号在宏语句中的含义后,对阅读内核的代码会带来帮助。