上節(jié)我們分析了C++基礎(chǔ)語(yǔ)法的const,static以及 this 指針,那么這節(jié)內(nèi)容我們來(lái)看一下 inline 內(nèi)聯(lián)函數(shù)吧!
inline 內(nèi)聯(lián)函數(shù)
特征
相當(dāng)于把內(nèi)聯(lián)函數(shù)里面的內(nèi)容寫(xiě)在調(diào)用內(nèi)聯(lián)函數(shù)處;
相當(dāng)于不用執(zhí)行進(jìn)入函數(shù)的步驟,直接執(zhí)行函數(shù)體;
相當(dāng)于宏,卻比宏多了類(lèi)型檢查,真正具有函數(shù)特性;
編譯器一般不內(nèi)聯(lián)包含循環(huán)、遞歸、switch 等復(fù)雜操作的內(nèi)聯(lián)函數(shù);
在類(lèi)聲明中定義的函數(shù),除了虛函數(shù)的其他函數(shù)都會(huì)自動(dòng)隱式地當(dāng)成內(nèi)聯(lián)函數(shù)。
inline 使用
// 聲明1(加 inline,建議使用)inline int functionName(int first, int second,。..);
// 聲明2(不加 inline)int functionName(int first, int second,。..);
// 定義inline int functionName(int first, int second,。..) {/****/};
// 類(lèi)內(nèi)定義,隱式內(nèi)聯(lián)class A { int doA() { return 0; } // 隱式內(nèi)聯(lián)}
// 類(lèi)外定義,需要顯式內(nèi)聯(lián)class A { int doA();}inline int A::doA() { return 0; } // 需要顯式內(nèi)聯(lián)
編譯器對(duì) inline 函數(shù)的處理步驟
將 inline 函數(shù)體復(fù)制到 inline 函數(shù)調(diào)用點(diǎn)處;
為所用 inline 函數(shù)中的局部變量分配內(nèi)存空間;
將 inline 函數(shù)的的輸入?yún)?shù)和返回值映射到調(diào)用方法的局部變量空間中;
如果 inline 函數(shù)有多個(gè)返回點(diǎn),將其轉(zhuǎn)變?yōu)?inline 函數(shù)代碼塊末尾的分支(使用 GOTO)。
優(yōu)缺點(diǎn)
優(yōu)點(diǎn)
內(nèi)聯(lián)函數(shù)同宏函數(shù)一樣將在被調(diào)用處進(jìn)行代碼展開(kāi),省去了參數(shù)壓棧、棧幀開(kāi)辟與回收,結(jié)果返回等,從而提高程序運(yùn)行速度。
內(nèi)聯(lián)函數(shù)相比宏函數(shù)來(lái)說(shuō),在代碼展開(kāi)時(shí),會(huì)做安全檢查或自動(dòng)類(lèi)型轉(zhuǎn)換(同普通函數(shù)),而宏定義則不會(huì)。
在類(lèi)中聲明同時(shí)定義的成員函數(shù),自動(dòng)轉(zhuǎn)化為內(nèi)聯(lián)函數(shù),因此內(nèi)聯(lián)函數(shù)可以訪(fǎng)問(wèn)類(lèi)的成員變量,宏定義則不能。
內(nèi)聯(lián)函數(shù)在運(yùn)行時(shí)可調(diào)試,而宏定義不可以。
缺點(diǎn)
代碼膨脹。內(nèi)聯(lián)是以代碼膨脹(復(fù)制)為代價(jià),消除函數(shù)調(diào)用帶來(lái)的開(kāi)銷(xiāo)。如果執(zhí)行函數(shù)體內(nèi)代碼的時(shí)間,相比于函數(shù)調(diào)用的開(kāi)銷(xiāo)較大,那么效率的收獲會(huì)很少。另一方面,每一處內(nèi)聯(lián)函數(shù)的調(diào)用都要復(fù)制代碼,將使程序的總代碼量增大,消耗更多的內(nèi)存空間。
inline 函數(shù)無(wú)法隨著函數(shù)庫(kù)升級(jí)而升級(jí)。inline函數(shù)的改變需要重新編譯,不像 non-inline 可以直接鏈接。
是否內(nèi)聯(lián),程序員不可控。內(nèi)聯(lián)函數(shù)只是對(duì)編譯器的建議,是否對(duì)函數(shù)內(nèi)聯(lián),決定權(quán)在于編譯器。
虛函數(shù)(virtual)可以是內(nèi)聯(lián)函數(shù)(inline)嗎?
虛函數(shù)可以是內(nèi)聯(lián)函數(shù),內(nèi)聯(lián)是可以修飾虛函數(shù)的,但是當(dāng)虛函數(shù)表現(xiàn)多態(tài)性的時(shí)候不能內(nèi)聯(lián)。
內(nèi)聯(lián)是在編譯器建議編譯器內(nèi)聯(lián),而虛函數(shù)的多態(tài)性在運(yùn)行期,編譯器無(wú)法知道運(yùn)行期調(diào)用哪個(gè)代碼,因此虛函數(shù)表現(xiàn)為多態(tài)性時(shí)(運(yùn)行期)不可以?xún)?nèi)聯(lián)。
inline virtual 唯一可以?xún)?nèi)聯(lián)的時(shí)候是:編譯器知道所調(diào)用的對(duì)象是哪個(gè)類(lèi)(如 Base::who()),這只有在編譯器具有實(shí)際對(duì)象而不是對(duì)象的指針或引用時(shí)才會(huì)發(fā)生。
虛函數(shù)內(nèi)聯(lián)使用
#include 《iostream》 using namespace std;class Base{public: inline virtual void who(){ cout 《《 “I am Base
”; } virtual ~Base() {}};class Derived : public Base{public: inline void who() // 不寫(xiě)inline時(shí)隱式內(nèi)聯(lián){ cout 《《 “I am Derived
”; }};
int main(){ // 此處的虛函數(shù) who(),是通過(guò)類(lèi)(Base)的具體對(duì)象(b)來(lái)調(diào)用的,編譯期間就能確定了,所以它可以是內(nèi)聯(lián)的,但最終是否內(nèi)聯(lián)取決于編譯器。 Base b; b.who();
// 此處的虛函數(shù)是通過(guò)指針調(diào)用的,呈現(xiàn)多態(tài)性,需要在運(yùn)行時(shí)期間才能確定,所以不能為內(nèi)聯(lián)。 Base *ptr = new Derived(); ptr-》who();
// 因?yàn)锽ase有虛析構(gòu)函數(shù)(virtual ~Base() {}),所以 delete 時(shí),會(huì)先調(diào)用派生類(lèi)(Derived)析構(gòu)函數(shù),再調(diào)用基類(lèi)(Base)析構(gòu)函數(shù),防止內(nèi)存泄漏。 delete ptr; ptr = nullptr;
system(“pause”); return 0;}
責(zé)任編輯:haq
-
函數(shù)
+關(guān)注
關(guān)注
3文章
4381瀏覽量
64895 -
C++
+關(guān)注
關(guān)注
22文章
2119瀏覽量
75324
原文標(biāo)題:C++基礎(chǔ)語(yǔ)法梳理:inline 內(nèi)聯(lián)函數(shù)!
文章出處:【微信號(hào):cyuyanxuexi,微信公眾號(hào):C語(yǔ)言編程學(xué)習(xí)基地】歡迎添加關(guān)注!文章轉(zhuǎn)載請(qǐng)注明出處。
發(fā)布評(píng)論請(qǐng)先 登錄
RealView MDK中使用內(nèi)聯(lián)函數(shù)時(shí)需要注意的問(wèn)題
C++標(biāo)準(zhǔn)編程:虛函數(shù)與內(nèi)聯(lián)
內(nèi)聯(lián)函數(shù)簡(jiǎn)介
IAR開(kāi)發(fā)環(huán)境默認(rèn)是不支持inline的嗎?
C++如何處理內(nèi)聯(lián)虛函數(shù)
內(nèi)聯(lián)函數(shù)詳解
內(nèi)聯(lián)函數(shù)和外聯(lián)函數(shù)有什么區(qū)別

C++語(yǔ)法的外圍基礎(chǔ)
C++基礎(chǔ)語(yǔ)法之inline 內(nèi)聯(lián)函數(shù)
深度解析C++中的虛函數(shù)

講解下C語(yǔ)言的內(nèi)聯(lián)函數(shù)
C語(yǔ)言內(nèi)聯(lián)函數(shù),提升C技巧必備
C語(yǔ)言內(nèi)聯(lián)函數(shù)

評(píng)論