# 的功能
# 的功能是將其后面的宏參數進行字符串化操作(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的特殊符號在宏語句中的含義后,對閱讀內核的代碼會帶來幫助。