日本免费高清视频-国产福利视频导航-黄色在线播放国产-天天操天天操天天操天天操|www.shdianci.com

學無先后,達者為師

網站首頁 編程語言 正文

C++inline函數的特性你了解嗎_C 語言

作者:guo5411 ? 更新時間: 2022-05-19 編程語言

一.inline的作用(內聯函數)

我們使用關鍵字inline和函數定義一起就可以創建一個內聯函數,它的作用就是減少函數調用的開銷,假如我們的程序中有一個函數會頻繁的被調用,這樣使程序的運行速度十分緩慢,那么我們使用內聯函數,就可以解決這種問題,提高程序的運行效率.

內聯函數的定義:

inline void add(int a, int b){
    return a+b;
}

上面這個就是一個內聯函數,假如我們的程序中頻繁使用add操作時,將其聲明為內聯函數就可以減少調用開銷提高效率.

注意:?

1.inline關鍵字是一種"用于實現的關鍵字",而不是"用于聲明的關鍵字", 換句話說inline只有放在函數的定義前,才有可能使該函數成為內聯函數,放在函數聲明前無效.

2.另外一點是:在c++的類中定義的成員函數默認就是內聯函數,但前提是在類中定義,因為我們正常的定義類的方式是在頭文件中聲明類的成員變量和成員函數,在原文件中對類的成員函數進行定義,此時成員函數的定義發生在類外,所以自然就不再是內聯函數了,如果還要讓其成為內聯函數,就需要我們手動添加inline.

3.至于第一點中我為什么要說可能使該函數成為內聯函數,是因為inline是一種推薦型關鍵字,他并不保證該函數一定會成為內聯函數,而是編譯器在編譯的時候獲取有inline建議的函數對其規模,內容,方法等要素進行判斷,如果滿足編譯器要求該函數會成為內聯函數,反之不會成為內聯函數,因此成不成內聯函數完全由編譯器說了算.

二.inline的實現(注意debug模式下內聯函數不會展開)

既然內聯函數可以減小函數的調用開銷,那么他是怎么實現的呢?非常簡單,我們在所有調用內聯函數的地方將其代碼展開,這樣雖然會使代碼的的量增加,但是減少了調用的開銷,是一種典型的空間換時間的做法.

我們將編譯器調到Release模式下使用內聯函數,在匯編指令下就能看見其實現的方式.

源碼:

#include
using namespace std;
inline int add(int a, int b) {
	return 0;
}
int main() {
	int b=add(1, 2);
	cout << b << endl;
	return 0;
}

(debug模式)反匯編:

在debug模式下我們可以清楚的看到,在程序運行到add函數時,進行了2和1兩個參數的壓棧,以及add函數的調用.

(Release模式)反匯編:

此時我們可以看到,像之前的那種參數壓棧,函數調用沒有了,那就是內聯函數,實現成功了,在這里不會再繼續調用add函數了,而是將add函數直接展開,將其代碼放在這里直接進行a+b的運算,避免函數的調用,增加其運行效率.

三.使用inline的注意事項

在上面我說到了,使用內聯函數是一種消耗空間獲得時間的方法,因此我們得把握好其中的度,當程序的規模過大時,其代碼的運行的開銷已經遠遠超過我們函數調用的開銷時,此時收益非常小,再使用內聯函數就沒有必要了,并且函數內存在遞歸或者循環時也不能使用內聯函數,編譯器也不會將其變為內聯函數.

源碼:

#include
using namespace std;
inline int mul(int a) {//求a的階乘
	if (a <= 0) {
		return 1;
	}
	return a * mul(a - 1);
}
int main() {
	int b=mul(10);
	cout << b << endl;
	return 0;
}

(Release模式)反匯編:

此時在Release模式的反匯編指令中我們看到mul函數并沒有被展開,而是進行了函數調用,說明編譯器并沒有將其變為內聯函數.

四.inline和預處理的區別

我們知道預處理階段會發生宏替換這一行為,宏替換也就是將代碼中用到宏的地方用宏定義的代碼或數據與宏進行替換,inline也是將函數中的代碼復制到調用內聯函數的地方,那么他們有什么區別呢?宏替換只是單純的替換并不安全,而內聯函數在使用的時候,編譯器會對內聯函數調用的正確性進行判斷如果不正確則該操作就不會發生,因此其更加安全,所以在c++中所有的宏代碼,都應該使用內聯函數進行取代.?

總結

原文鏈接:https://blog.csdn.net/weixin_49312527/article/details/123503198

欄目分類
最近更新