C語言中單井號(#)和雙井號(##)在宏語句中的應用
在閱讀Linux內核代碼過程中,特別是一些預處理指令宏的時候,會看到宏語句里會包含一些# 或者是連著的## 符號,剛接觸的時候覺得很一頭霧水,但這些宏語句有時候繞不開,所以為了更好地讀懂這些代碼,很有必要仔細學些一下這些特殊符號的含義。
# 的功能
# 的功能是將其后面的宏參數(shù)進行字符串化操作(stringfication),簡單說就是在對它所引用的宏變量通過替換后在其左右各加上一個雙引號。
1. #define WARN_IF(exp)
2. do {
3. if (exp)
4. fprintf(stderr, "Warning: " #exp "n")?
5. } while(0)
現(xiàn)在在程序中以下面的方式調用這個宏:
1. WARN_IF (divider == 0)?
那么在編譯時,上面的這句話被擴展為:
1. do { if (divider == 0) fprintf(stderr, "Warning: " "divider == 0" "n")? } while(0)?
這樣每次divider(除數(shù))為0的時候便會在標準錯誤流上輸出一個提示信息。
## 的功能
## 稱為連接符號(concatenator),由兩個# 號組成,其功能是在帶參數(shù)的宏定義中將兩個子串(token)聯(lián)接起來,從而形成一個新的子串。但它不可以是第一個或者最后一個子串。所謂的子串(token)就是指編譯器能夠識別的最小語法單元,注意這里連接的對象是token就行,而不一定是宏的變量。
下面舉個例子來看看它們是怎樣工作的。假設程序中已經(jīng)定義了這樣一個帶參數(shù)的宏:
1. #define LINK_MULTIPLE(a,b,c,d) a##_##b##_##c##_##d
現(xiàn)在在程序中以下面的方式調用這個宏:
1. struct _record_type LINK_MULTIPLE(name,company,position,salary)?
那么在編譯時,上面的這句話被擴展為:
1. struct _record_type name_company_position_salary?
綜合舉例
最后舉一個綜合# 和## 的例子。假設程序中已經(jīng)定義了這樣一個帶參數(shù)的宏:
1. #define paster( n ) printf( "token" #n " = %d", token##n )
現(xià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
相信大家理解了這些特殊符號在宏語句中的含義后,對閱讀內核的代碼會帶來幫助。





