C++深入分析內(nèi)聯(lián)函數(shù)的使用
一、常量與宏回顧
C++中的const常量可以替代宏常數(shù)定義,如︰
但是C++中是否有解決方替代宏代碼片段呢?這里就要引入內(nèi)聯(lián)函數(shù)。
二、內(nèi)聯(lián)函數(shù)
- C++中推薦使用內(nèi)聯(lián)函數(shù)替代宏代碼片段
- C++中使用 inline 關(guān)鍵字聲明內(nèi)聯(lián)函數(shù)
內(nèi)聯(lián)函數(shù)聲明時(shí)inline關(guān)鍵字必須和函數(shù)定義結(jié)合在一起,否則編譯器會(huì)直接忽略內(nèi)聯(lián)請(qǐng)求
- C++編譯器可以將一個(gè)函數(shù)進(jìn)行內(nèi)聯(lián)編譯
- 被C++編譯器內(nèi)聯(lián)編譯的函數(shù)叫做內(nèi)聯(lián)函數(shù)
- C++編譯器直接將函數(shù)體插入函數(shù)調(diào)用的地方
- 內(nèi)聯(lián)函數(shù)沒有普通函數(shù)調(diào)用時(shí)的額外開銷(壓棧,跳轉(zhuǎn),返回)
C++編譯器不一定滿足函數(shù)的內(nèi)聯(lián)請(qǐng)求
下面先看一段宏定義的代碼:
#include <stdio.h> #define FUNC(a, b) ((a) < (b) ? (a) : (b)) inline int func(int a, int b) { return a < b ? a : b; } int main(int argc, char *argv[]) { int a = 1; int b = 3; int c = FUNC(++a, b); printf("a = %d\n", a); printf("b = %d\n", b); printf("c = %d\n", c); return 0; }
下面為輸出結(jié)果,c =FUNC(++a, b),等價(jià)于 c =((++a) < (b) ? (++a) : (b)),先執(zhí)行(++a) < (b),得到 a =,2,b = 3,由于 ++a 比 b 小,所以輸出 ++a,這個(gè)時(shí)候 a = 3,所以 c 也等于 3。
如果使用內(nèi)聯(lián)函數(shù),
#include <stdio.h> #define FUNC(a, b) ((a) < (b) ? (a) : (b)) inline int func(int a, int b) { return a < b ? a : b; } int main(int argc, char *argv[]) { int a = 1; int b = 3; int c = func(++a, b); printf("a = %d\n", a); printf("b = %d\n", b); printf("c = %d\n", c); return 0; }
輸出結(jié)果如下:
下面在 VS2012 中看一下反匯編的代碼,看一看內(nèi)聯(lián)有沒有成功。可以看到 func 函數(shù)被調(diào)用,而不是直接擴(kuò)展到被調(diào)用的地方,內(nèi)聯(lián)沒有成功。inline 這個(gè)關(guān)鍵字僅僅是請(qǐng)求將函數(shù)內(nèi)聯(lián),但是不一定會(huì)成功。
為了讓編譯器允許對(duì)內(nèi)聯(lián)的請(qǐng)求,可以對(duì)編譯器進(jìn)行一些配置,如下圖
配置好以后,在 int c = func(++a, b); 前面打個(gè)斷點(diǎn),開始跑代碼,但是我跑的時(shí)候報(bào)錯(cuò)了。
這個(gè)時(shí)候,進(jìn)行下面配置,那個(gè)錯(cuò)誤就解決了。
這個(gè)時(shí)候在進(jìn)行反匯編,如下圖所示,可以看到調(diào)用函數(shù)的匯編代碼沒有了,表示內(nèi)聯(lián)成功了。
- 內(nèi)聯(lián)函數(shù)具有普通函數(shù)的特征(參數(shù)檢查,返回類型等)
- 函數(shù)的內(nèi)聯(lián)請(qǐng)求可能被編譯器拒絕
- 函數(shù)被內(nèi)聯(lián)編譯后,函數(shù)體直接擴(kuò)展到調(diào)用的地方
宏代碼片段由預(yù)處理器處理,進(jìn)行簡單的文本替換,沒有任何編譯過程,因此可能出現(xiàn)副作用。
現(xiàn)代C++編譯器能夠進(jìn)行編譯優(yōu)化,一些函數(shù)即使沒有inline聲明,也可能被內(nèi)聯(lián)編譯
一些現(xiàn)代C++編譯器提供了擴(kuò)展語法,能夠?qū)瘮?shù)進(jìn)行強(qiáng)制內(nèi)聯(lián),如︰
- g++ : _attribute__((always_inline)) 屬性
- MSVC : _forceinline
三、內(nèi)聯(lián)函數(shù)使用注意事項(xiàng)
C++中 inline 內(nèi)聯(lián)編譯的限制:
- 不能存在任何形式的循環(huán)語句
- 不能存在過多的條件判斷語句
- 函數(shù)體不能過于龐大
- 不能對(duì)函數(shù)進(jìn)行取址操作
- 不能對(duì)函數(shù)進(jìn)行取址操作
四、小結(jié)
- C++中可以通過 inline 聲明內(nèi)聯(lián)函數(shù)
- 編譯器直接將內(nèi)聯(lián)函數(shù)體擴(kuò)展到函數(shù)調(diào)用的地方
- inline 只是一種請(qǐng)求,編譯器不一定允許這種請(qǐng)求
- 內(nèi)聯(lián)函數(shù)省去了函數(shù)調(diào)用時(shí)壓棧,跳轉(zhuǎn)和返回的開銷
到此這篇關(guān)于C++深入分析內(nèi)聯(lián)函數(shù)的使用的文章就介紹到這了,更多相關(guān)C++內(nèi)聯(lián)函數(shù)內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
C語言編程銀行ATM存取款系統(tǒng)實(shí)現(xiàn)源碼
這篇文章主要為大家介紹了C語言編程銀行ATM存取款系統(tǒng)實(shí)現(xiàn)的源碼示例,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步2021-11-11C++使用expected實(shí)現(xiàn)優(yōu)雅的錯(cuò)誤處理
C++ 中提供了很多中方式進(jìn)行錯(cuò)誤處理。無論是通過拋異常還是通過錯(cuò)誤碼,標(biāo)準(zhǔn)庫都提供相應(yīng)的調(diào)用,今天本文為大家介紹的是使用expected進(jìn)行錯(cuò)誤處理,感興趣的可以了解一下2023-06-06C++迭代器介紹(iterator、const_iterator、reverse_interator、const_rev
這篇文章主要介紹了C++迭代器介紹(iterator、const_iterator、reverse_interator、const_reverse_interator),文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-02-02