定義的意思
創(chuàng)新互聯(lián)公司-專(zhuān)業(yè)網(wǎng)站定制、快速模板網(wǎng)站建設(shè)、高性?xún)r(jià)比洛寧網(wǎng)站開(kāi)發(fā)、企業(yè)建站全套包干低至880元,成熟完善的模板庫(kù),直接使用。一站式洛寧網(wǎng)站制作公司更省心,省錢(qián),快速模板網(wǎng)站建設(shè)找我們,業(yè)務(wù)覆蓋洛寧地區(qū)。費(fèi)用合理售后完善,10余年實(shí)體公司更值得信賴(lài)。
在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
等等
宏定義的基礎(chǔ)知識(shí)。引用宏定義時(shí),直接代入進(jìn)行代換。
既然已經(jīng)宏定義SUB(a) (a)-(a),而程序中出現(xiàn)的對(duì)應(yīng)a的是a+b,那么就將a換為a+b代入表達(dá)式:
d=SUB(a+b)*c=(a+b)-(a+b)*c=(2+3)-(2+3)*5
直接用(a+b)-(a+b)代換SUB(a+b)。這一點(diǎn)和數(shù)學(xué)是不同的,不要強(qiáng)行往數(shù)學(xué)上靠。
可以把宏理解成拼字游戲,它功能很強(qiáng)大,但是強(qiáng)大到使用不好就會(huì)有副作用。C++有很多語(yǔ)言設(shè)施用來(lái)完全特定功能的宏,如const,inline,template,就是為了讓大家少用宏。給你舉個(gè)宏和函數(shù)不同的例子代碼:
#define max(x,y) ((x)(y)?(x):(y))
template class T
inline T max(T x,T y){return xy?x:y;}
看起來(lái)似乎是相同的功能,可是函數(shù)調(diào)用,畢竟會(huì)求完每一個(gè)實(shí)參的值,再傳遞給被調(diào)函數(shù),即使聲明了inline,在調(diào)用點(diǎn)展開(kāi)而不發(fā)生實(shí)際的調(diào)用開(kāi)銷(xiāo)。
但是你試試用這個(gè)調(diào)用宏,結(jié)果就會(huì)有問(wèn)題:
int i=4,j=5;
int k=max(i++,j++);
如果是函數(shù)調(diào)用,i==5,j==6,k==5。如果是宏的話(huà),結(jié)果是:
int k=((i++)(j++)?(i++):(j++));
你覺(jué)得會(huì)一樣嗎?所以,慎用宏。
MFC中有很多功能是宏完成的,它太強(qiáng)大了,很多情況下有宏很高效,但是不容易控制。
宏是用于編譯器處理的,他在程序編譯時(shí),會(huì)在對(duì)應(yīng)位置展開(kāi)成代碼。。,這就相當(dāng)于你在告訴編譯器,我想在這個(gè)位置加一些代碼,代碼的內(nèi)容已在宏中定義,請(qǐng)編譯器自己支找。。。,也就是說(shuō)程序在運(yùn)行時(shí),早已變成了對(duì)應(yīng)位置上的代碼,此時(shí)已沒(méi)有宏的概念了。。。。
而函數(shù)則是運(yùn)行時(shí),調(diào)用。他不會(huì)在編譯時(shí),在對(duì)應(yīng)位置上加上函數(shù)代碼,只是加上一個(gè)函數(shù)入口指針。。。從這個(gè)入口去運(yùn)行一段代碼。。。運(yùn)行完了之后回到當(dāng)前位置繼續(xù)執(zhí)行。。。。
可以簡(jiǎn)單的認(rèn)為,宏是在編譯時(shí)上起作用,而函數(shù)是運(yùn)行時(shí)起作用。。。
宏是一種預(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)行條件編譯的用法與使用宏防止多重引用類(lèi)似。示例如下:
使用條件編譯,方便程序員在調(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)化還依賴(lài)于編譯器的實(shí)現(xiàn),而使用宏函數(shù)則完全由代碼本身控制。
但在使用宏函數(shù)的時(shí)候,需要明確的是宏函數(shù)只是簡(jiǎn)單的替換,需要注意括號(hào)的使用。
擴(kuò)展資料:
宏的更多規(guī)則特性:
(1)宏名一般用大寫(xiě)。
(2)使用宏可提高程序的通用性和易讀性,減少不一致性,減少輸入錯(cuò)誤和便于修改。例如:數(shù)組大小常用宏定義。
(3)預(yù)處理是在編譯之前的處理,而編譯工作的任務(wù)之一就是語(yǔ)法檢查,預(yù)處理不做語(yǔ)法檢查。
(4)宏定義末尾不加分號(hào)。
(5)宏定義寫(xiě)在函數(shù)的花括號(hào)外邊,作用域?yàn)槠浜蟮某绦?,通常在文件的最開(kāi)頭。
(6)可以用#undef命令終止宏定義的作用域。
(7)宏定義不可以嵌套。
(8)字符串" "中永遠(yuǎn)不包含宏。
(9)宏定義不分配內(nèi)存,變量定義分配內(nèi)存。
(10)宏定義不存在類(lèi)型問(wèn)題,它的參數(shù)也是無(wú)類(lèi)型的。
參考資料:
百度百科--宏定義
函數(shù)式宏(function-like macro)較之對(duì)象式宏可以進(jìn)行更復(fù)雜的代換。
函數(shù)式宏 sqr 是在編譯時(shí)展開(kāi)并填入程序的,因此 只要是能用雙目運(yùn)算符 * 進(jìn)行乘法計(jì)算的數(shù)據(jù)類(lèi)型,都能使用函數(shù)式宏 。
函數(shù)定義則需為每個(gè)形參都定義各自的數(shù)據(jù)類(lèi)型,返回值的類(lèi)型也只能為一種。就這點(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é)果,我們稱(chēng)這種情況為宏的 副作用(side effect) 。