inline內聯函式

2021-06-13 17:27:23 字數 2986 閱讀 4947

技術類筆試題50%都會問巨集與inline的區別,自己去找找看?

1)巨集替換發生在預編譯

2)巨集函式(如果可以這麼叫的話)替換時不會檢查引數,inline函式會檢查

3)巨集一定會發生替換,inline貌似不是強制的,編譯器想不替換也沒關係

4)巨集替換時存在著一些不可避免的陷阱(參見c traps and pitfalls),例如傳參時如果傳了a++之類的可能會出錯,inline就比較安全了。

巨集有***,比如max(x++,y++) inline會不同,

慎用內聯內聯能提高函式的執行效率,為什麼不把所有的函式都定義成內聯函式?如果所有的函式都是內聯函式,還用得著「內聯」這個關鍵字嗎?內聯是以**膨脹(複製)為代價,僅僅省去了函式呼叫的開銷,從而提高函式的執行效率。如果執行函式體內**的時間,相比於函式呼叫的開銷較大,那麼效率的收穫會很少。另一方面,每一處內聯函式的呼叫都要複製**,將使程式的總**量增大,消耗更多的記憶體空間。以下情況不宜使用內聯:(1)如果函式體內的**比較長,使用內聯將導致記憶體消耗代價較高。(2)如果函式體內出現迴圈,那麼執行函式體內**的時間要比函式呼叫的開銷大。類的建構函式和析構函式容易讓人誤解成使用內聯更有效。要當心建構函式和析構函式可能會隱藏一些行為,如「偷偷地」執行了基類或成員物件的建構函式和析構函式。所以不要隨便地將建構函式和析構函式的定義體放在類宣告中。乙個好的編譯器將會根據函式的定義體,自動地取消不值得的內聯(這進一步說明了inline 不應該出現在函式的宣告中)。

介紹內聯函式之前,有必要介紹一下預處理巨集。內聯函式的功能和預處理巨集的功能相似。相信大家都用過預處理巨集,我們會經常定義一些巨集,如

#define table_comp(x) ((x)>0?(x):0)

就定義了乙個巨集。

為什麼要使用巨集呢?因為函式的呼叫必須要將程式執行的順序轉移到函式所存放在記憶體中的某個位址,將函式的程式內容執行完後,再返回到轉去執行該函式前的地方。這種轉移操作要求在轉去執行前要儲存現場並記憶執行的位址,轉回後要恢復現場,並按原來儲存位址繼續執行。因此,函式呼叫要有一定的時間和空間方面的開銷,於是將影響其效率。而巨集只是在預處理的地方把**展開,不需要額外的空間和時間方面的開銷,所以呼叫乙個巨集比呼叫乙個函式更有效率。

但是巨集也有很多的不盡人意的地方。

1、.巨集不能訪問物件的私有成員。

2、.巨集的定義很容易產生二意性。

我們舉個例子:

#define table_multi(x) (x*x)

我們用乙個數字去呼叫它,table_multi(10),這樣看上去沒有什麼錯誤,結果返回100,是正確的,但是如果我們用table_multi(10+10)去呼叫的話,我們期望的結果是400,而巨集的呼叫結果是(10+10*10+10),結果是120,這顯然不是我們要得到的結果。避免這些錯誤的方法,一是給巨集的引數都加上括號。

#define table_multi(x) ((x)*(x))

這樣可以確保不會出錯,但是,即使使用了這種定義,這個巨集依然有可能出錯,例如使用table_multi(a++)呼叫它,他們本意是希望得到(a+1)*(a+1)的結果,而實際上呢?我們可以看看巨集的展開結果:(a++)*(a++)

,如果a的值是4,我們得到的結果是5*6=30。而我們期望的結果是5*5=25,這又出現了問題。事實上,在一些c的庫函式中也有這些問題。例如: toupper(*pchar++)就會對pchar執行兩次++操作,因為toupper實際上也是乙個巨集。

我們可以看到巨集有一些難以避免的問題,怎麼解決呢?

下面就是用我要介紹的內聯函式來解決這些問題,我們可以使用內聯函式來取代巨集的定義。而且事實上我們可以用內聯函式完全取代預處理巨集。

內聯函式和巨集的區別在於,巨集是由預處理器對巨集進行替代,而內聯函式是通過編譯器控制來實現的。而且內聯函式是真正的函式,只是在需要用到的時候,內聯函式像巨集一樣的展開,所以取消了函式的引數壓棧,減少了呼叫的開銷。你可以象呼叫函式一樣來呼叫內聯函式,而不必擔心會產生於處理巨集的一些問題。

我們可以用inline來定義內聯函式,不過,任何在類的說明部分定義的函式都會被自動的認為是內聯函式。

下面我們來介紹一下內聯函式的用法。

內聯函式必須是和函式體申明在一起,才有效。像這樣的申明inline tablefunction(int i)是沒有效果的,編譯器只是把函式作為普通的函式申明,我們必須定義函式體。

inline tablefunction(int i) ;

這樣我們才算定義了乙個內聯函式。我們可以把它作為一般的函式一樣呼叫。但是執行速度確比一般函式的執行速度要快。

我們也可以將定義在類的外部的函式定義為內聯函式,比如:

class tableclass;

inline int dec()

int getnum();}

inline int tableclass::getnum()

上面申明的三個函式都是內聯函式。在c++中,在類的內部定義了函式體的函式,被預設為是內聯函式。而不管你是否有inline關鍵字。

內聯函式在c++類中,應用最廣的,應該是用來定義訪問函式。我們定義的類中一般會把資料成員定義成私有的或者保護的,這樣,外界就不能直接讀寫我們類成員的資料了。對於私有或者保護成員的讀寫就必須使用成員介面函式來進行。如果我們把這些讀寫成員函式定義成內聯函式的話,將會獲得比較好的效率。

class sample

void settest(int i) }

當然,內聯函式也有一定的侷限性。就是函式中的執行**不能太多了,如果,內聯函式的函式體過大,一般的編譯器會放棄內聯方式,而採用普通的方式呼叫函式。這樣,內聯函式就和普通函式執行效率一樣了。

#define expressionname(var1,var2) (var1+var2)*(var1-var2)

這種表示式形式巨集形式與作用跟函式類似,但它使用預編譯器,沒有堆疊,使用上比函式高效。但它只是預編譯器上符號表的簡單替換,不能進行引數有效性檢測及使用c++類的成員訪問控制。

inline 推出的目的,也正是為了取代這種表示式形式的巨集定義,它消除了它的缺點,同時又很好地繼承了它的優點。inline**放入預編譯器符號表中,高效;它是個真正的函式,呼叫時有嚴格的引數檢測;它也可作為類的成員函式。

inline 內聯函式

一 inline關鍵字用來定義乙個類的內聯函式,引入它的主要原因是用它替代c中表示式形式的巨集定義。表示式形式的巨集定義一例 define expressionname var1,var2 var1 var2 var1 var2 為什麼要取代這種形式呢,且聽我道來 1 首先談一下在c中使用這種形式巨...

inline內聯函式

一 內聯函式 內聯函式是指用inline關鍵字修飾的函式 它與普通函式所不同之處只在於函式呼叫的處理。普通 函式進行呼叫時,要將程式執行權轉到被呼叫函式中,然後再返回到呼叫它的函式中 內聯函式不是在呼叫時發生執行權轉移,而是在編譯時將 函式 體 嵌入在每乙個呼叫處。編譯時類似巨集替換,使用 函式體替...

inline內聯函式

demo 帶引數的巨集 define myfunc a,b a b a b inline int myfunc int a,int b int main1402 說明1 必須inline int myfunc int a,int b 和函式體的實現,寫在一塊 說明2c 編譯器可以將乙個函式進行內聯編...