內(nèi)聯(lián)函數(shù)是C語(yǔ)言從C++中借鑒過(guò)來(lái)的,適當(dāng)?shù)氖褂脙?nèi)聯(lián)函數(shù)可以提高程序的執(zhí)行效率。本篇文章就來(lái)講解下內(nèi)聯(lián)函數(shù),趕緊來(lái)看下吧!
一、函數(shù)調(diào)用
在講內(nèi)聯(lián)函數(shù)之前,我們需要先了解函數(shù)調(diào)用,而函數(shù)調(diào)用,又不得不說(shuō)函數(shù)調(diào)用的開(kāi)銷。
一個(gè)函數(shù)執(zhí)行的時(shí)候,經(jīng)常會(huì)調(diào)用另一個(gè)函數(shù),比如執(zhí)行函數(shù)A時(shí),我們需要對(duì)一些數(shù)據(jù)進(jìn)行處理,將運(yùn)算結(jié)果暫存在R0寄存器,接著要調(diào)用另一個(gè)函數(shù)B,而函數(shù)B也用到了R0這個(gè)寄存器(用于保存函數(shù)的返回值),原本函數(shù)A暫存在R0寄存器的值就被改變了,這樣做肯定不行。
現(xiàn)代計(jì)算機(jī)系統(tǒng)的做法都是會(huì)在執(zhí)行函數(shù)B之前,先把R0寄存器的值保存到堆棧中,函數(shù)B執(zhí)行結(jié)束后,再將堆棧中的值恢復(fù)到R0寄存器中,然后函數(shù)A繼續(xù)執(zhí)行,這樣對(duì)于數(shù)據(jù)處理就不會(huì)有任何問(wèn)題了。
但是,函數(shù)調(diào)用卻消耗一定的時(shí)間進(jìn)行切換,這段時(shí)間用來(lái)保存現(xiàn)場(chǎng)和恢復(fù)現(xiàn)場(chǎng),大約相當(dāng)于一兩條語(yǔ)句的執(zhí)行時(shí)間,這就是函數(shù)調(diào)用帶來(lái)的開(kāi)銷。
假如函數(shù)B很小,只有一兩行代碼,從上圖我們可以看出,真正只有函數(shù)B執(zhí)行代碼的那段時(shí)間是對(duì)我們有用的,切換帶來(lái)的就是額外的成本開(kāi)銷了,如果函數(shù)A里面多次調(diào)用函數(shù)B,那開(kāi)銷就更明顯了。
二、內(nèi)聯(lián)函數(shù)
函數(shù)B很小,又被頻繁的調(diào)用,可能函數(shù)調(diào)用的切換時(shí)間比函數(shù)內(nèi)代碼的執(zhí)行時(shí)間還長(zhǎng),這樣明顯劃不來(lái),那么我們就可以將這個(gè)函數(shù)聲明為內(nèi)聯(lián)(加上 inline),編譯器在編譯時(shí),會(huì)把內(nèi)聯(lián)函數(shù)的實(shí)現(xiàn)替換到每個(gè)調(diào)用內(nèi)聯(lián)函數(shù)的地方(可以與宏函數(shù)做類比),在調(diào)用處將代碼展開(kāi),相當(dāng)于自動(dòng)將函數(shù)B的代碼在調(diào)用它的地方復(fù)制了一份副本,沒(méi)有了保護(hù)現(xiàn)場(chǎng)和恢復(fù)現(xiàn)場(chǎng)的時(shí)間,從而節(jié)省了函數(shù)調(diào)用的開(kāi)銷。
內(nèi)聯(lián)函數(shù)一般要求如下:
1. 函數(shù)體積小,通常5行以內(nèi);
2. 被頻繁調(diào)用;
3. 函數(shù)內(nèi)無(wú)復(fù)雜的實(shí)現(xiàn),比如:while、for循環(huán),switch,遞歸等;
4. 函數(shù)沒(méi)有包含靜態(tài)變量。
來(lái)看一個(gè)簡(jiǎn)單的內(nèi)聯(lián)函數(shù)的例子:
#includemain函數(shù)代碼在執(zhí)行的時(shí)候是這樣的://將函數(shù)max_value聲明為inline inlineintmax_value(intx,inty) { return(x>y)?x:y; } intmain() { inta=1,b=2; intm; m=max_value(a,b); return0; }
intmain() { inta=1,b=2; intm; m=(1>2)?1:2; return0; }內(nèi)聯(lián)函數(shù)在調(diào)用處展開(kāi)了。
在c++ 中定義在類里面的函數(shù),默認(rèn)情況下都是內(nèi)聯(lián)的,比如下面這種情況:
#includeusingnamespacestd; classHunTalk_Linux { public: //默認(rèn)是內(nèi)聯(lián)函數(shù) intmax_value(intx,inty) { return(x>y)?x:y; } }; intmain() { return0; }
注意:函數(shù)聲明為內(nèi)聯(lián),僅僅是對(duì)編譯器的建議,如果函數(shù)比較復(fù)雜,編譯器會(huì)將其看做普通函數(shù)。
三、內(nèi)聯(lián)函數(shù)與宏
前面講到可以與宏函數(shù)做類比,那么就納悶了,為什么不直接定義一個(gè)宏,而是定義一個(gè)內(nèi)聯(lián)函數(shù)?存在即合理,自然有它存在的道理,相對(duì)于宏,內(nèi)聯(lián)函數(shù)提供了更好的方法:
參數(shù)類型檢查。編譯過(guò)程中,宏調(diào)用并不執(zhí)行類型檢查,甚至連正常參數(shù)也不檢查,內(nèi)聯(lián)函數(shù)雖然具有宏的展開(kāi)特性,但其本質(zhì)仍是函數(shù),編譯器仍可以對(duì)其進(jìn)行參數(shù)檢查,而宏就不具備這個(gè)功能。
在宏中的編譯錯(cuò)誤很難發(fā)現(xiàn),因?yàn)樗鼈円玫氖菙U(kuò)展的代碼,而不是程序員鍵入的。
便于調(diào)試。內(nèi)聯(lián)函數(shù)代碼的調(diào)試信息通常比擴(kuò)展的宏代碼更有用,它同樣可以支持?jǐn)帱c(diǎn)、單步......等調(diào)試功能。
接口封裝。有些內(nèi)聯(lián)函數(shù)可以用來(lái)封裝一個(gè)接口,而宏不具備這個(gè)特性。
四、總結(jié)
引入內(nèi)聯(lián)函數(shù)主要是解決一些頻繁調(diào)用的小函數(shù)造成額外時(shí)間開(kāi)銷的問(wèn)題,但是也要在符合一定內(nèi)聯(lián)函數(shù)的情況下使用。
使用很多的內(nèi)聯(lián)函數(shù),每個(gè)調(diào)用該函數(shù)的地方都需要替換成函數(shù)體,代碼量就會(huì)增加,代碼量就會(huì)增加也同時(shí)帶來(lái)了潛在的編譯時(shí)間的增加。
算法里面有個(gè)概念叫空間換時(shí)間,就是使用內(nèi)存占用更大的算法換取執(zhí)行速度的提升,所以說(shuō)適當(dāng)?shù)氖褂脙?nèi)聯(lián)函數(shù)可以提高程序的執(zhí)行效率。
審核編輯:湯梓紅
-
計(jì)算機(jī)
+關(guān)注
關(guān)注
19文章
7163瀏覽量
87126 -
C語(yǔ)言
+關(guān)注
關(guān)注
180文章
7575瀏覽量
133975 -
C++
+關(guān)注
關(guān)注
21文章
2084瀏覽量
73297 -
內(nèi)聯(lián)函數(shù)
+關(guān)注
關(guān)注
0文章
10瀏覽量
2199 -
函數(shù)調(diào)用
+關(guān)注
關(guān)注
0文章
19瀏覽量
2571
原文標(biāo)題:C語(yǔ)言內(nèi)聯(lián)函數(shù),提升C技巧必備
文章出處:【微信號(hào):混說(shuō)Linux,微信公眾號(hào):混說(shuō)Linux】歡迎添加關(guān)注!文章轉(zhuǎn)載請(qǐng)注明出處。
發(fā)布評(píng)論請(qǐng)先 登錄
相關(guān)推薦
評(píng)論