1、如何區(qū)分宏定義中的“宏名稱”和“宏字符串”?對(duì)于帶參數(shù)的宏又該注意什么?
在宏定義中,“宏名稱”和“宏字符串”是通過(guò)“空格”來(lái)區(qū)分的。編譯器在處理時(shí)宏定義時(shí),首先從“#define”后第一個(gè)空格開(kāi)始讀取字符串,直到遇見(jiàn)下一個(gè)空格為止,兩個(gè)空格之間的字符串為“宏名稱”,確定好“宏名稱”之后,本行的所有其他字符串都為“宏字符串”。圖示:#define + N個(gè)空格(1
對(duì)于“帶參數(shù)宏”,宏名稱和“( )”之間不能有空格,否則就變成了“無(wú)參數(shù)宏”(根據(jù)上面的原則)。而且當(dāng)“無(wú)參數(shù)宏”和“帶參數(shù)宏”的名字相同時(shí),“無(wú)參數(shù)宏”會(huì)屏蔽掉“帶參數(shù)宏”,即使以“帶參數(shù)宏”的方式調(diào)用,也行不通。
試驗(yàn)內(nèi)容及結(jié)果:
實(shí)驗(yàn)分析:
可以看到“#define PI ?3.1415”和“#define P I 689”分別是兩個(gè)不同的宏定義“PI”和“P”;"G(4)"被“(X) (2*X) (4)”替換掉;當(dāng)調(diào)用“F(4)”時(shí),系統(tǒng)并沒(méi)有替換成“2*4”,而是替換成了“123(4)”,說(shuō)明“#define F ?123”完全屏蔽掉了“#define F(X) ?(2*X)”,當(dāng)注釋掉“#define ?F ?123”后"#define F(X) ?(2*X)"可以正常工作。因此,在進(jìn)行宏定義時(shí),要密切關(guān)注空格的影響,并且“帶參數(shù)宏”和“無(wú)參數(shù)宏”的名稱一定不能相同,否則會(huì)出現(xiàn)混亂。但是,在宏調(diào)用時(shí)空格并不影響效果,例如F(3)和F (3)效果相同(F(X)是帶參數(shù)宏)。
2、宏和函數(shù)在使用方式和效果上有何異同?
在宏定義時(shí),要善于利用括號(hào)對(duì)變量進(jìn)行封裝,把每個(gè)參數(shù)都括起來(lái),預(yù)防出現(xiàn)與優(yōu)先級(jí)有關(guān)的問(wèn)題;同時(shí)整個(gè)結(jié)果表達(dá)式也應(yīng)該括起來(lái),以防止當(dāng)宏用于一個(gè)更復(fù)雜的表達(dá)式時(shí)可能出現(xiàn)問(wèn)題。盡量提高宏的可靠性,。例如:“#define ABS(x) ?(((x) > 0) ? (x) : (-x))”的可靠性要遠(yuǎn)遠(yuǎn)優(yōu)于“#define ABS(x) ?x > 0 ? x : -x”,可以以ABS(a-b)來(lái)進(jìn)行試驗(yàn)。
在宏調(diào)用時(shí),如果有自增(++)或自減(--)操作符,一定要注意很可能會(huì)產(chǎn)生副作用。因?yàn)楹暝谔鎿Q時(shí),如果變量出現(xiàn)了多次,就相當(dāng)于自增或自減操作進(jìn)了多次,這個(gè)跟函數(shù)調(diào)用是完全不同的,函數(shù)調(diào)用中形參會(huì)復(fù)制實(shí)參的數(shù)值,并對(duì)形參進(jìn)行操作并不會(huì)影響實(shí)參,而宏調(diào)用就是直接多次修改實(shí)參。例如:a = 5; "ABS(a++) “展開(kāi)后就變成“(((a++) > 0) ? (a++) : (-a++))”,操作完成后”a = 7“而不是”a = 6“;當(dāng)寫成函數(shù)就完全不用擔(dān)心這個(gè)問(wèn)題。
如果在宏調(diào)用時(shí),進(jìn)行了了多層嵌套調(diào)用,則宏展開(kāi)后會(huì)產(chǎn)生非常龐大的表達(dá)式,而且相當(dāng)復(fù)雜;函數(shù)調(diào)用則不會(huì)出現(xiàn)這種情況。
3、宏和類型定義typedef的區(qū)別
由于宏的本質(zhì)就是替換,所以可以對(duì)變量類型進(jìn)行一層封裝,利用該封裝做變量定義,這樣做的好處是增加可移植性,當(dāng)修改時(shí)只需要改動(dòng)宏定義即可。例如:
但是最好不要這么用,因?yàn)槲覀冇衪ypedef,它是專門進(jìn)行類型定義的。而且,使用類型定義會(huì)使代碼更加通用一些,避免一些深層的問(wèn)題。例如:
?
?
分析:
?
從概念上看,MY_TYPE1 和 MY_TYPE2 完全相同,都是指向uint_8的指針,但是當(dāng)我們聲明多個(gè)變量時(shí),就出現(xiàn)問(wèn)題了。它們分別被擴(kuò)展成了:
可以看到,本來(lái)想定義兩個(gè)指針變量a,b;現(xiàn)在卻變成了一個(gè)指針變量a和一個(gè)整型變量b,這不是我們想要的。而MY_TYPE2本身就是一種類型(自定義)了,故c,d都是指針類型,符合預(yù)期。所以,如果想自定義類型,果斷選擇 ”typedef“ 放棄宏定義,否則吃虧的是自己。
以上就是關(guān)于C語(yǔ)言中宏定義的使用分析的全部?jī)?nèi)容,感謝大家的閱讀,更多內(nèi)容請(qǐng)關(guān)注武林技術(shù)頻道網(wǎng)站。
|
新聞熱點(diǎn)
疑難解答
圖片精選