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