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 關鍵字聲明內(nèi)聯(lián)函數(shù)

內(nèi)聯(lián)函數(shù)聲明時inline關鍵字必須和函數(shù)定義結合在一起,否則編譯器會直接忽略內(nèi)聯(lián)請求
- C++編譯器可以將一個函數(shù)進行內(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)用時的額外開銷(壓棧,跳轉,返回)
C++編譯器不一定滿足函數(shù)的內(nèi)聯(lián)請求
下面先看一段宏定義的代碼:
#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;
}下面為輸出結果,c =FUNC(++a, b),等價于 c =((++a) < (b) ? (++a) : (b)),先執(zhí)行(++a) < (b),得到 a =,2,b = 3,由于 ++a 比 b 小,所以輸出 ++a,這個時候 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;
}輸出結果如下:

下面在 VS2012 中看一下反匯編的代碼,看一看內(nèi)聯(lián)有沒有成功??梢钥吹?func 函數(shù)被調(diào)用,而不是直接擴展到被調(diào)用的地方,內(nèi)聯(lián)沒有成功。inline 這個關鍵字僅僅是請求將函數(shù)內(nèi)聯(lián),但是不一定會成功。

為了讓編譯器允許對內(nèi)聯(lián)的請求,可以對編譯器進行一些配置,如下圖


配置好以后,在 int c = func(++a, b); 前面打個斷點,開始跑代碼,但是我跑的時候報錯了。

這個時候,進行下面配置,那個錯誤就解決了。

這個時候在進行反匯編,如下圖所示,可以看到調(diào)用函數(shù)的匯編代碼沒有了,表示內(nèi)聯(lián)成功了。

- 內(nèi)聯(lián)函數(shù)具有普通函數(shù)的特征(參數(shù)檢查,返回類型等)
- 函數(shù)的內(nèi)聯(lián)請求可能被編譯器拒絕
- 函數(shù)被內(nèi)聯(lián)編譯后,函數(shù)體直接擴展到調(diào)用的地方
宏代碼片段由預處理器處理,進行簡單的文本替換,沒有任何編譯過程,因此可能出現(xiàn)副作用。
現(xiàn)代C++編譯器能夠進行編譯優(yōu)化,一些函數(shù)即使沒有inline聲明,也可能被內(nèi)聯(lián)編譯
一些現(xiàn)代C++編譯器提供了擴展語法,能夠對函數(shù)進行強制內(nèi)聯(lián),如︰
- g++ : _attribute__((always_inline)) 屬性
- MSVC : _forceinline
三、內(nèi)聯(lián)函數(shù)使用注意事項
C++中 inline 內(nèi)聯(lián)編譯的限制:
- 不能存在任何形式的循環(huán)語句
- 不能存在過多的條件判斷語句
- 函數(shù)體不能過于龐大
- 不能對函數(shù)進行取址操作
- 不能對函數(shù)進行取址操作
四、小結
- C++中可以通過 inline 聲明內(nèi)聯(lián)函數(shù)
- 編譯器直接將內(nèi)聯(lián)函數(shù)體擴展到函數(shù)調(diào)用的地方
- inline 只是一種請求,編譯器不一定允許這種請求
- 內(nèi)聯(lián)函數(shù)省去了函數(shù)調(diào)用時壓棧,跳轉和返回的開銷
到此這篇關于C++深入分析內(nèi)聯(lián)函數(shù)的使用的文章就介紹到這了,更多相關C++內(nèi)聯(lián)函數(shù)內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!
相關文章
C++實現(xiàn)LeetCode(29.兩數(shù)相除)
這篇文章主要介紹了C++實現(xiàn)LeetCode(29.兩數(shù)相除),本篇文章通過簡要的案例,講解了該項技術的了解與使用,以下就是詳細內(nèi)容,需要的朋友可以參考下2021-07-07
C語言動態(tài)內(nèi)存泄露常見問題內(nèi)存分配改進方法詳解
今天遇見了一道有意思的內(nèi)存泄露題目,特地分享給大家,相信屏幕前的你學習完一定有所收獲,預祝讀者學習愉快,多多進步早日升職加薪2021-10-10

