C++中的inline關鍵字
1. 引入inline關鍵字的原因
在c/c++中,為了解決一些頻繁呼叫的小函式大量消耗棧空間(棧記憶體)的問題,特別的引入了inline修飾符,表示為行內函數。
棧空間就是指放置程式的區域性資料(也就是函式內資料)的記憶體空間。
在系統下,棧空間是有限的,假如頻繁大量的使用就會造成因棧空間不足而導致程式出錯的問題,如,函式的死迴圈遞迴呼叫的最終結果就是導致棧記憶體空間枯竭。
下面我們來看一個例子:
#include <stdio.h>
//函式定義為inline即:行內函數
inline char* dbtest(int a) {
return (i % 2 > 0) ? "奇" : "偶";
}
int main()
{
int i = 0;
for (i=1; i < 100; i++) {
printf("i:%d 奇偶性:%s /n", i, dbtest(i));
}
}
上面的例子就是標準的行內函數的用法,使用inline修飾帶來的好處我們表面看不出來,其實,在內部的工作就是在每個for迴圈的內部任何呼叫dbtest(i)的地方都換成了(i%2>0)?”奇”:”偶”,這樣就避免了頻繁呼叫函式對棧記憶體重複開闢所帶來的消耗。
2. inline使用限制
inline的使用是有所限制的
3. inline僅是一個對編譯器的建議
inline函式僅僅是一個對編譯器的建議,所以最後能否真正內聯,看編譯器的意思,它如果認為函式不復雜,能在呼叫點展開,就會真正內聯,並不是說聲明瞭內聯就會內聯,宣告內聯只是一個建議而已。
4. 建議:inline函式的定義放在標頭檔案中
其次,因為行內函數要在呼叫點展開,所以編譯器必須隨處可見行內函數的定義,要不然就成了非行內函數的呼叫了。所以,這要求每個呼叫了行內函數的檔案都出現了該行內函數的定義
因此,將行內函數的定義放在標頭檔案裡實現是合適的,省卻你為每個檔案實現一次的麻煩。
宣告跟定義要一致:如果在每個檔案裡都實現一次該行內函數的話,那麼,最好保證每個定義都是一樣的,否則,將會引起未定義的行為。如果不是每個檔案裡的定義都一樣,那麼,編譯器展開的是哪一個,那要看具體的編譯器而定。所以,最好將行內函數定義放在標頭檔案中。
5. 類中的成員函式與inline
定義在類中的成員函式預設都是內聯的,如果在類定義時就在類內給出函式定義,那當然最好。如果在類中未給出成員函式定義,而又想內聯該函式的話,那在類外要加上inline,否則就認為不是內聯的。
例如,
class A
{
public:void Foo(int x, int y) { } // 自動地成為行內函數
}
將成員函式的定義體放在類宣告之中雖然能帶來書寫上的方便,但不是一種良好的程式設計風格,上例應該改成:
// 標頭檔案
class A
{
public:
void Foo(int x, int y);
}
// 定義檔案
inline void A::Foo(int x, int y){}
6. inline 是一種“用於實現的關鍵字”
關鍵字inline 必須與函式定義體放在一起才能使函式成為內聯,僅將inline 放在函式宣告前面不起任何作用。
如下風格的函式Foo 不能成為行內函數:
inline void Foo(int x, int y); // inline 僅與函式宣告放在一起
void Foo(int x, int y){}
而如下風格的函式Foo 則成為行內函數:
void Foo(int x, int y);
inline void Foo(int x, int y) {} // inline 與函式定義體放在一起
所以說,inline 是一種“用於實現的關鍵字”,而不是一種“用於宣告的關鍵字”。一般地,使用者可以閱讀函式的宣告,但是看不到函式的定義。儘管在大多數教科書中行內函數的宣告、定義體前面都加了inline 關鍵字,但我認為inline不應該出現在函式的宣告中。這個細節雖然不會影響函式的功能,但是體現了高質量C++/C 程式設計風格的一個基本原則:宣告與定義不可混為一談,使用者沒有必要、也不應該知道函式是否需要內聯。
7. 慎用inline
內聯能提高函式的執行效率,為什麼不把所有的函式都定義成行內函數?如果所有的函式都是行內函數,還用得著“內聯”這個關鍵字嗎?
內聯是以程式碼膨脹(複製)為代價,僅僅省去了函式呼叫的開銷,從而提高函式的執行效率。
如果執行函式體內程式碼的時間,相比於函式呼叫的開銷較大,那麼效率的收穫會很少。另一方面,每一處行內函數的呼叫都要複製程式碼,將使程式的總程式碼量增大,消耗更多的記憶體空間。
以下情況不宜使用內聯:
(1)如果函式體內的程式碼比較長,使用內聯將導致記憶體消耗代價較高。
(2)如果函式體內出現迴圈,那麼執行函式體內程式碼的時間要比函式呼叫的開銷大。類的建構函式和解構函式容易讓人誤解成使用內聯更有效。要當心建構函式和解構函式可能會隱藏一些行為,如“偷偷地”執行了基類或成員物件的建構函式和解構函式。所以不要隨便地將建構函式和解構函式的定義體放在類宣告中。一個好的編譯器將會根據函式的定義體,自動地取消不值得的內聯(這進一步說明了 inline 不應該出現在函式的宣告中)。
8.總結
行內函數並不是一個增強效能的靈丹妙藥。只有當函式非常短小的時候它才能得到我們想要的效果;但是,如果函式並不是很短而且在很多地方都被呼叫的話,那麼將會使得可執行體的體積增大。
最令人煩惱的還是當編譯器拒絕內聯的時候。在老的實現中,結果很不盡人意,雖然在新的實現中有很大的改善,但是仍然還是不那麼完善的。一些編譯器能夠足夠的聰明來指出哪些函式可以內聯哪些不能,但是大多數編譯器就不那麼聰明瞭,因此這就需要我們的經驗來判斷。如果行內函數不能增強效能,就避免使用它!
相關推薦
C/C++中extern關鍵字詳解
編譯器 fin 生成 接口 bcd 只需要 c++環境 結束 編程 轉自:http://www.cnblogs.com/yc_sunniwell/archive/2010/07/14/1777431.html 1 基本解釋:extern可以置於變量或者函數前,以標示變量或者
c#中params關鍵字應用
語言 bsp for arr als new 傳參數 lin long c#params應用 params 是C#開發語言中關鍵字, params主要的用處是在給函數傳參數的時候用,就是當函數的參數不固定的時候。 在方法聲明中的 params 關鍵字之後不允許任何其他參數,
C/C++中const關鍵字的用法及其與宏定義的比較
類型安全 屬性 const關鍵字 code oid 程序 函數返回值 存儲空間 臨時對象 1.const關鍵字的性質 簡單來說:const關鍵字修飾的變量具有常屬性。 即它所修飾的變量不能被修改。 2.修飾局部變量 1 const int a = 10; 2 int co
C++中static關鍵字作用總結
可見性 apple 局部變量 內存 父類靜態 初始化 ati 退出 變量存儲 1.先來介紹它的第一條也是最重要的一條:隱藏。(static函數,static變量均可) 當同時編譯多個文件時,所有未加static前綴的全局變量和函數都具有全局可見性。舉例來說明。同時編譯兩個源
淺析C++中static關鍵字
參數 extern 函數 iostream 出現 har 兩個 什麽 註意事項 C++的static有兩種用法:面向過程程序設計中的static和面向對象程序設計中的static。前者應用於普通變量和函數,不涉及類;後者主要說明static在類中的作用。 一、面向過程設計中
C/C++中inline/static inline/extern inline的區別及使用
分享一下我老師大神的人工智慧教程!零基礎,通俗易懂!http://blog.csdn.net/jiangjunshow 也歡迎大家轉載本篇文章。分享知識,造福人民,實現我們中華民族偉大復興!  
c++中extern關鍵字和static的作用域的學習
之前對static修飾變數後作用域的問題一直是大體瞭解,感覺和普通變數的作用域並沒有什麼區別。聚個栗子 在A.h中: #pragma once static int a=365; class A { public: A(); ~A(); }; 在main.c中: #include
C/C++中volatile關鍵字詳解
asm 運行 多線程並發 這樣的 修改 由於 設定 其他 硬件 1. 為什麽用volatile? C/C++ 中的 volatile 關鍵字和 const 對應,用來修飾變量,通常用於建立語言級別的 memory barrier。這是 BS 在 "The C++ P
C/C++中 static關鍵字
C 語言的 static 關鍵字有三種(具體來說是兩種)用途: 1. 靜態區域性變數:用於函式體內部修飾變數,這種變數的生存期長於該函式。 int foo(){ static int i = 1; // note:1 //int i = 1
C++中static關鍵字的作用
static的作用主要有兩個方面: 限定作用域; 保持變數內容持久化。 static關鍵字在C++中的用法: 只在cpp內有效的全域性變數: 在cpp檔案的全域性範圍內宣告: static int val = 0; 這個變數的含義是該cpp內有效,
C++中const關鍵字詳解
const關鍵字作用 1. 修飾變數 用法:const 型別說明符 變數名。 含義:說明該變數不可以被改變。 用途:常量命名等 2. 修飾
C++中const關鍵字修飾
C++中const使用的地方非常多,修飾包括:全域性變數、函式、函式引數等等,這篇文章主要是總結一下const各種使用的規則以及需要注意事項。 const名叫常量限定符,用來限定特定變數,以通知編譯器該變數是不可修改的。習慣性的使用const,可以避免在函式中對某些不應
C++中static關鍵字作用
@著作權歸作者所有:來自CSDN部落格作者大鬍子的艾娃的原創作品,如需轉載,請註明出處,否則將追究法律責任。 如有錯誤的地方歡迎指正,謝謝! 一、 static宣告全域性靜態變數和區域性靜態變數 1、static宣告全域性靜態變數 該全域性變數只能供本模組使用,
C++中const關鍵字增強
.c檔案 #include<stdio.h> const int m_a = 10; //全域性作用域的const不能使用指標進行修改 int main() { const int m_b = 10; int *p = (int *)&m_b; *p =
C++中explicit關鍵字的作用
explicit用來防止由建構函式定義的隱式轉換。 要明白它的作用,首先要了解隱式轉換:可以用單個實參來呼叫的建構函式定義了從形參型別到該類型別的一個隱式轉換。 例如:
C和C#中static關鍵字的作用
1. static 變數 靜態變數的型別說明符是static。靜態變數當然是屬於靜態儲存方式,但是屬於靜態儲存方式的量不一定就是靜態變數。例如外部變數雖屬於靜態儲存方式,但不一定是靜態變數,必須由static加以定義後才能成為靜態外部變數,或稱靜態全域性變數。2. 靜態區域性變數 靜態區域性變數屬於靜態儲
C++中auto關鍵字的使用及編譯錯誤解決
auto可以自動幫我們推斷變數的型別 比如在定義一些難以確定的變數型別時 35 func_log(__func__, ""); 36 // vector<Student>::iterator it = St
C#中this關鍵字的使用
this的兩個作用: 1、代表當前類的物件; 2、在類當中顯示的呼叫本類的建構函式 例如以下兩個建構函式中: public Students(string name,char gender,i
c#中volatile關鍵字的作用(zz)
恐怕比較一下volatile和synchronized的不同是最容易解釋清楚的。volatile是變 量修飾符,而synchronized則作用於一段程式碼或方法;看如下三句get程式碼: int i1; int geti1() {return
C#中Internal關鍵字的總結
注意:想要徹底把Internal關鍵字搞清楚,就耐著性子把她讀完。當然了這篇文章只是對其他文章的總結。也算是引用吧。主要還是為了把知識點搞清楚 進入主題之前先來了解一下,專案、解決方案、程式集、名稱空間四個容易混淆的概念。①專案:就是我們開發的一個軟體。.NET下,專案有多