C語言宏定義使用分析
1、如何區(qū)分宏定義中的“宏名稱”和“宏字符串”?對于帶參數(shù)的宏又該注意什么?
在宏定義中,“宏名稱”和“宏字符串”是通過“空格”來區(qū)分的。編譯器在處理時宏定義時,首先從“#define”后第一個空格開始讀取字符串,直到遇見下一個空格為止,兩個空格之間的字符串為“宏名稱”,確定好“宏名稱”之后,本行的所有其他字符串都為“宏字符串”。圖示:#define + N個空格(1 < N) + 宏名稱(中間沒有空格) + N個空格(1 < N) + 宏字符串(直到本行結(jié)束)。這里講到的都是一行之內(nèi)的宏定義,如果跨越多行則用“\”字符進(jìn)行“續(xù)行”,本質(zhì)上可以當(dāng)做一行來對待。
對于“帶參數(shù)宏”,宏名稱和“( )”之間不能有空格,否則就變成了“無參數(shù)宏”(根據(jù)上面的原則)。而且當(dāng)“無參數(shù)宏”和“帶參數(shù)宏”的名字相同時,“無參數(shù)宏”會屏蔽掉“帶參數(shù)宏”,即使以“帶參數(shù)宏”的方式調(diào)用,也行不通。
試驗內(nèi)容及結(jié)果:
實驗分析:
可以看到“#define PI 3.1415”和“#define P I 689”分別是兩個不同的宏定義“PI”和“P”;"G(4)"被“(X) (2*X) (4)”替換掉;當(dāng)調(diào)用“F(4)”時,系統(tǒng)并沒有替換成“2*4”,而是替換成了“123(4)”,說明“#define F 123”完全屏蔽掉了“#define F(X) (2*X)”,當(dāng)注釋掉“#define F 123”后"#define F(X) (2*X)"可以正常工作。因此,在進(jìn)行宏定義時,要密切關(guān)注空格的影響,并且“帶參數(shù)宏”和“無參數(shù)宏”的名稱一定不能相同,否則會出現(xiàn)混亂。但是,在宏調(diào)用時空格并不影響效果,例如F(3)和F (3)效果相同(F(X)是帶參數(shù)宏)。
2、宏和函數(shù)在使用方式和效果上有何異同?
在宏定義時,要善于利用括號對變量進(jìn)行封裝,把每個參數(shù)都括起來,預(yù)防出現(xiàn)與優(yōu)先級有關(guān)的問題;同時整個結(jié)果表達(dá)式也應(yīng)該括起來,以防止當(dāng)宏用于一個更復(fù)雜的表達(dá)式時可能出現(xiàn)問題。盡量提高宏的可靠性,。例如:“#define ABS(x) (((x) > 0) ? (x) : (-x))”的可靠性要遠(yuǎn)遠(yuǎn)優(yōu)于“#define ABS(x) x > 0 ? x : -x”,可以以ABS(a-b)來進(jìn)行試驗。
在宏調(diào)用時,如果有自增(++)或自減(--)操作符,一定要注意很可能會產(chǎn)生副作用。因為宏在替換時,如果變量出現(xiàn)了多次,就相當(dāng)于自增或自減操作進(jìn)了多次,這個跟函數(shù)調(diào)用是完全不同的,函數(shù)調(diào)用中形參會復(fù)制實參的數(shù)值,并對形參進(jìn)行操作并不會影響實參,而宏調(diào)用就是直接多次修改實參。例如:a = 5; "ABS(a++) “展開后就變成“(((a++) > 0) ? (a++) : (-a++))”,操作完成后”a = 7“而不是”a = 6“;當(dāng)寫成函數(shù)就完全不用擔(dān)心這個問題。
如果在宏調(diào)用時,進(jìn)行了了多層嵌套調(diào)用,則宏展開后會產(chǎn)生非常龐大的表達(dá)式,而且相當(dāng)復(fù)雜;函數(shù)調(diào)用則不會出現(xiàn)這種情況。
3、宏和類型定義typedef的區(qū)別
由于宏的本質(zhì)就是替換,所以可以對變量類型進(jìn)行一層封裝,利用該封裝做變量定義,這樣做的好處是增加可移植性,當(dāng)修改時只需要改動宏定義即可。例如:
MY_TYPE a;
MY_TYPE b,c,d;
但是最好不要這么用,因為我們有typedef,它是專門進(jìn)行類型定義的。而且,使用類型定義會使代碼更加通用一些,避免一些深層的問題。例如:
typedef uint_8 * MY_TYPE2
MY_TYPE1 a,b;
MY_TYPE2 c,d;
從概念上看,MY_TYPE1 和 MY_TYPE2 完全相同,都是指向uint_8的指針,但是當(dāng)我們聲明多個變量時,就出現(xiàn)問題了。它們分別被擴(kuò)展成了:
uint_8 *a,b;
MY_TYPE2 c,d; //因為MY_TYPE2已經(jīng)是一種類型了
可以看到,本來想定義兩個指針變量a,b;現(xiàn)在卻變成了一個指針變量a和一個整型變量b,這不是我們想要的。而MY_TYPE2本身就是一種類型(自定義)了,故c,d都是指針類型,符合預(yù)期。所以,如果想自定義類型,果斷選擇 ”typedef“ 放棄宏定義,否則吃虧的是自己。
相關(guān)文章
C++深入探索類真正的形態(tài)之struct與class
前邊我們所定義的類,均是使用struct關(guān)鍵字來定義,但是C++中真正用于定義類的關(guān)鍵字為class,因為要C++兼容C,所以保留struct關(guān)鍵字,struct與class的用法完全相同2022-04-04簡要對比C語言中的truncate()函數(shù)與ftruncate()函數(shù)
這篇文章主要介紹了C語言中的truncate()函數(shù)與ftruncate()函數(shù)的簡要對比,注意其之間的區(qū)別,需要的朋友可以參考下2015-09-09C語言數(shù)學(xué)公式來實現(xiàn)土味表白
大家好,本篇文章主要講的是C語言數(shù)學(xué)公式來實現(xiàn)土味表白,感興趣的同學(xué)趕快來看一看吧,對你有幫助的話記得收藏一下,方便下次瀏覽2021-12-12HDOJ 1443 約瑟夫環(huán)的最新應(yīng)用分析詳解
本篇文章是對HDOJ 1443 約瑟夫環(huán)的最新應(yīng)用進(jìn)行了詳細(xì)的分析介紹,需要的朋友參考下2013-05-05