函數(shù)式宏(function-like macro)較之對(duì)象式宏可以進(jìn)行更復(fù)雜的代換。
創(chuàng)新互聯(lián)為企業(yè)提供:品牌網(wǎng)站制作、網(wǎng)絡(luò)營(yíng)銷策劃、小程序開(kāi)發(fā)、營(yíng)銷型網(wǎng)站建設(shè)和網(wǎng)站運(yùn)營(yíng)托管,一站式網(wǎng)絡(luò)營(yíng)銷整體服務(wù)。實(shí)現(xiàn)不斷獲取潛在客戶之核心目標(biāo),建立了企業(yè)專屬的“全網(wǎng)整合營(yíng)銷推廣”,就用不著再為了獲取潛在客戶而苦惱,相反,客戶會(huì)主動(dòng)找您,生意就找上門來(lái)了!
函數(shù)式宏 sqr 是在編譯時(shí)展開(kāi)并填入程序的,因此 只要是能用雙目運(yùn)算符 * 進(jìn)行乘法計(jì)算的數(shù)據(jù)類型,都能使用函數(shù)式宏 。
函數(shù)定義則需為每個(gè)形參都定義各自的數(shù)據(jù)類型,返回值的類型也只能為一種。就這點(diǎn)而言,函數(shù)較為嚴(yán)格。
函數(shù)為我們默默無(wú)聞地進(jìn)行了一些復(fù)雜處理,如:
而函數(shù)式宏所做的工作只是宏展開(kāi)和填入程序,并不進(jìn)行上述處理。
根據(jù)以上特征, 函數(shù)式宏或許能使程序的運(yùn)行速度稍微提高一點(diǎn),但是程序自身卻有可能變得臃腫 。
函數(shù)式宏在使用上必須小心謹(jǐn)慎。例如, sqr(a++) 展開(kāi)后 ((a++) * (a++)) 。每次展開(kāi), a 的值都會(huì)自增兩次。在不經(jīng)意間表達(dá)式被執(zhí)行了兩次,導(dǎo)致程序出現(xiàn)預(yù)料之外的結(jié)果,我們稱這種情況為宏的 副作用(side effect) 。
定義的意思
在C語(yǔ)言中的宏定義就是定義外部變量 內(nèi)部函數(shù)使用exiterm來(lái)調(diào)用
如:exiterm char c1,c2;
C語(yǔ)言中的宏定義的字是:define
C語(yǔ)言中的define宏定義有帶參數(shù)的和不帶參數(shù)兩中形式
不帶參格式是:#define 宏名 字符串
帶參格式是:#define 宏名(參數(shù)1,參數(shù)2,...) 字符串
在C語(yǔ)言里宏定義只用來(lái)做的宏名替換,而不做語(yǔ)法檢查的,因而它不是C語(yǔ)句,所以在宏定義的時(shí)候不需要在后面加";"
宏也在C里也叫預(yù)處理命令,因?yàn)楹晔窃诔绦蚓幾g前先進(jìn)行字符替換的,所以叫預(yù)處理.在C里還有其它的預(yù)處理命令如:
#define
#undef
#ifdef
等等
宏是一種預(yù)處理指令,它提供了一種機(jī)制,可以用來(lái)替換源代碼中的字符串。
1、條件編譯:
C語(yǔ)言中,預(yù)處理過(guò)程讀入源代碼,檢查包含預(yù)處理指令的語(yǔ)句和宏定義,并對(duì)源代碼進(jìn)行相應(yīng)的轉(zhuǎn)換,預(yù)處理過(guò)程還會(huì)刪除程序中的注釋和多余的空白符號(hào)。
預(yù)處理指令是以#開(kāi)頭的代碼行,#必須是該行除了空白字符外的第一個(gè)字符。#后是指令關(guān)鍵字,在#和指令關(guān)鍵字之間允許存在若干空白字符。
使用宏進(jìn)行條件編譯的用法與使用宏防止多重引用類似。示例如下:
使用條件編譯,方便程序員在調(diào)試程序的過(guò)程中,執(zhí)行一些在程序發(fā)布后并不需要執(zhí)行的指令。只要在需要調(diào)試的代碼前加上_DEBUG的定義,就可以在調(diào)試程序的過(guò)程中輸出調(diào)試信息。
這樣方便查看程序在運(yùn)行過(guò)程中有沒(méi)有出現(xiàn)錯(cuò)誤,定位錯(cuò)誤出現(xiàn)的地方。而在程序發(fā)布之前,取消_DEBUG的定義就可以不再執(zhí)行調(diào)試代碼。
2、宏函數(shù):
函數(shù)的調(diào)用是需要一定的時(shí)間和空間代價(jià)的。因?yàn)橄到y(tǒng)在調(diào)用函數(shù)時(shí),需要保留"現(xiàn)場(chǎng)",即將程序要執(zhí)行的指令的下一條指令的位置壓入棧,然后轉(zhuǎn)入調(diào)用函數(shù)去執(zhí)行,調(diào)用完函數(shù)后再返回主調(diào)函數(shù),恢復(fù)"現(xiàn)場(chǎng)",返回到棧里保存的的下一條指令的位置繼續(xù)執(zhí)行。
所以函數(shù)的調(diào)用需要額外的時(shí)間和空間代價(jià)。
而宏函數(shù)則不存在上述問(wèn)題,宏函數(shù)在預(yù)編譯時(shí),同函數(shù)定義的代碼來(lái)替換函數(shù)名,將函數(shù)代碼段嵌入到當(dāng)前程序,不會(huì)產(chǎn)生函數(shù)調(diào)用。
所以會(huì)省去普通函數(shù)保留現(xiàn)場(chǎng)恢復(fù)現(xiàn)場(chǎng)的時(shí)間,但因?yàn)橐獙⒍x的函數(shù)體嵌入到當(dāng)前程序,所以不可避免的會(huì)占用額外的存儲(chǔ)空間。
在頻繁調(diào)用同一個(gè)宏的時(shí)候,該現(xiàn)象尤其明顯。宏函數(shù)的示例定義如下:
#define MAX(a,b) ((a)(b)?(b):(a))
宏函數(shù)的優(yōu)點(diǎn)在于避免函數(shù)調(diào)用,提高程序效率。
同時(shí)需要注意的是inline標(biāo)識(shí)符。inline也將函數(shù)定義為內(nèi)聯(lián)的。但是使用內(nèi)聯(lián)函數(shù)需要注意的是:函數(shù)體必須十分簡(jiǎn)單,不能含有循環(huán)、條件、選擇等復(fù)雜結(jié)構(gòu),否則就不能作為內(nèi)聯(lián)函數(shù)了。
事實(shí)上,有時(shí)候即便你沒(méi)有將函數(shù)指定為內(nèi)聯(lián)函數(shù),編譯器也會(huì)將一些簡(jiǎn)單的函數(shù)作為內(nèi)聯(lián)函數(shù)處理,而對(duì)于一些復(fù)雜的函數(shù),即使聲明為內(nèi)聯(lián)函數(shù),編譯器也不會(huì)理會(huì)的。
inline函數(shù)的瓶頸就在于此,使用inline標(biāo)識(shí)符將函數(shù)聲明為內(nèi)聯(lián)的,但這只是一種提示,到底編譯器有沒(méi)有優(yōu)化還依賴于編譯器的實(shí)現(xiàn),而使用宏函數(shù)則完全由代碼本身控制。
但在使用宏函數(shù)的時(shí)候,需要明確的是宏函數(shù)只是簡(jiǎn)單的替換,需要注意括號(hào)的使用。
擴(kuò)展資料:
宏的更多規(guī)則特性:
(1)宏名一般用大寫。
(2)使用宏可提高程序的通用性和易讀性,減少不一致性,減少輸入錯(cuò)誤和便于修改。例如:數(shù)組大小常用宏定義。
(3)預(yù)處理是在編譯之前的處理,而編譯工作的任務(wù)之一就是語(yǔ)法檢查,預(yù)處理不做語(yǔ)法檢查。
(4)宏定義末尾不加分號(hào)。
(5)宏定義寫在函數(shù)的花括號(hào)外邊,作用域?yàn)槠浜蟮某绦?,通常在文件的最開(kāi)頭。
(6)可以用#undef命令終止宏定義的作用域。
(7)宏定義不可以嵌套。
(8)字符串" "中永遠(yuǎn)不包含宏。
(9)宏定義不分配內(nèi)存,變量定義分配內(nèi)存。
(10)宏定義不存在類型問(wèn)題,它的參數(shù)也是無(wú)類型的。
參考資料:
百度百科--宏定義
直接調(diào)用即可。如下示例:
#includestdio.h#define sum(b,c){int d=0;d=b+c;printf("兩者的和:%d\n",d);}int main(){ int x=0,y=0; scanf("%d%d",x,y); sum (x,y); return 0;}使用宏函數(shù)或者宏函數(shù)時(shí),注意如下幾點(diǎn):
(1)宏函數(shù)的參數(shù)沒(méi)有類型,預(yù)處理器只負(fù)責(zé)做形式上的替換,而不做參數(shù)類型檢查,所以傳參時(shí)要格外小心。
(2)宏函數(shù)定義要注意格式,尤其是括號(hào)。
如果上面的宏函數(shù)寫成 #define MAX(a, b) (ab?a:b),省去內(nèi)層括號(hào),則宏展開(kāi)就成了k = (i0x0fj0x0f?i0x0f:j0x0f),運(yùn)算的優(yōu)先級(jí)就錯(cuò)了。同樣道理,這個(gè)宏定義的外層括號(hào)也是不能省的。若函數(shù)中是宏替換為 ++MAX(a,b),則宏展開(kāi)就成了 ++(a)(b)?(a):(b),運(yùn)算優(yōu)先級(jí)也是錯(cuò)了。
(3)宏函數(shù)往往會(huì)導(dǎo)致較低的代碼執(zhí)行效率。
int a[]={9,3,5,2,1,0,8,7,6,4};int max(n){return n==0?a[0]:MAX(a[n],max(n-1));}int main(){max(9);return 0;}若是普通函數(shù),則通過(guò)遞歸,可取的最大值,時(shí)間復(fù)雜度為O(n)。但若是宏函數(shù),則宏展開(kāi)為( a[n]max(n-1)?a[n]:max(n-1) ),其中max(n-1)被調(diào)用了兩遍,這樣依此遞歸下去,時(shí)間復(fù)雜度會(huì)很高。