內(nèi)聯(lián)函數(shù)詳解
什么是內(nèi)聯(lián)性和外聯(lián)函數(shù)
類的成員函數(shù)可以分為內(nèi)聯(lián)函數(shù)和外聯(lián)函數(shù)。內(nèi)聯(lián)函數(shù)是指那些定義在類體內(nèi)的成員函數(shù),即該函數(shù)的函數(shù)體放在類體內(nèi)。而說明在類體內(nèi),定義在類體外的成員函數(shù)叫外聯(lián)函數(shù)。外聯(lián)函數(shù)的函數(shù)體在類的實現(xiàn)部分。
內(nèi)聯(lián)函數(shù)在調(diào)用時不是像一般的函數(shù)那樣要轉(zhuǎn)去執(zhí)行被調(diào)用函數(shù)的函數(shù)體,執(zhí)行完成后再轉(zhuǎn)回調(diào)用函數(shù)中,執(zhí)行其后語句,而是在調(diào)用函數(shù)處用內(nèi)聯(lián)函數(shù)體的代碼來替換,這樣將會節(jié)省調(diào)用開銷,提高運行速度。
內(nèi)聯(lián)函數(shù)與前面講過的帶參數(shù)的宏定義進行一下比較,它們的代碼效率是一樣的,但是內(nèi)聯(lián)函數(shù)要優(yōu)于宏定義,因為內(nèi)聯(lián)函數(shù)遵循函數(shù)的類型和作用域規(guī)則,它與一般函數(shù)更相近,在一些編譯器中,一旦關(guān)上內(nèi)聯(lián)擴展,將與一般函數(shù)一樣進行調(diào)用,調(diào)試比較方便。
外聯(lián)函數(shù)變成內(nèi)聯(lián)函數(shù)的方法很簡單,只要在函數(shù)頭前面加上關(guān)鍵字inline就可以了。
#include iostream>
using namespace std;
class A
{
public:
A(int x, int y) //內(nèi)聯(lián)函數(shù)
{
X=x;Y=y;
}
int a() //內(nèi)聯(lián)函數(shù)
{
return X;
}
int b() //內(nèi)聯(lián)函數(shù)
{
return Y;
}
int c();
int d();
private:
int X,Y;
};
//inline定義內(nèi)聯(lián)函數(shù)
inline int A::c()
{
return a()+b();
}
inline int A::d()
{
return c();
}
void main()
{
A m(3,5);
int I=m.d();
coutd()return:Iendl;
}
輸出結(jié)果:
d()return:8
說明:類A中,直接定義了3個內(nèi)聯(lián)函數(shù),又使用inline定義了2個內(nèi)聯(lián)函數(shù)。
引入內(nèi)聯(lián)函數(shù)的意義
函數(shù)是一種更高級的抽象。它的引入使得編程者只關(guān)心函數(shù)的功能和使用方法,而不必關(guān)心函數(shù)功能的具體實現(xiàn);函數(shù)的引入可以減少程序的目標代碼,實現(xiàn)程序代碼和數(shù)據(jù)的共享。但是,函數(shù)調(diào)用也會帶來降低效率的問題,因為調(diào)用函數(shù)實際上將程序執(zhí)行順序轉(zhuǎn)移到函數(shù)所存放在內(nèi)存中某個地址,將函數(shù)的程序內(nèi)容執(zhí)行完后,再返回到轉(zhuǎn)去執(zhí)行該函數(shù)前的地方。這種轉(zhuǎn)移操作要求在轉(zhuǎn)去前要保護現(xiàn)場并記憶執(zhí)行的地址,轉(zhuǎn)回后先要恢復現(xiàn)場,并按原來保存地址繼續(xù)執(zhí)行。因此,函數(shù)調(diào)用要有一定的時間和空間方面的開銷,于是將影響其效率。特別是對于一些函數(shù)體代碼不是很大,但又頻繁地被調(diào)用的函數(shù)來講,解決其效率問題更為重要。引入內(nèi)聯(lián)函數(shù)實際上就是為了解決這一問題。
在程序編譯時,編譯器將程序中出現(xiàn)的內(nèi)聯(lián)函數(shù)的調(diào)用表達式用內(nèi)聯(lián)函數(shù)的函數(shù)體來進行替換。顯然,這種做法不會產(chǎn)生轉(zhuǎn)去轉(zhuǎn)回的問題,但是由于在編譯時將函數(shù)體中的代碼被替代到程序中,因此會增加目標程序代碼量,進而增加空間開銷,而在時間代銷上不象函數(shù)調(diào)用時那么大,可見它是以目標代碼的增加為代價來換取時間的節(jié)省。
在程序中,調(diào)用其函數(shù)時,該函數(shù)在編譯時被替代,而不是像一般函數(shù)那樣是在運行時被調(diào)用。
使用內(nèi)聯(lián)函數(shù)應注意的事項
內(nèi)聯(lián)函數(shù)具有一般函數(shù)的特性,它與一般函數(shù)所不同之處只在于函數(shù)調(diào)用的處理。一般函數(shù)進行調(diào)用時,要將程序執(zhí)行權(quán)轉(zhuǎn)到被調(diào)用函數(shù)中,然后再返回到調(diào)用它的函數(shù)中;而內(nèi)聯(lián)函數(shù)在調(diào)用時,是將調(diào)用表達式用內(nèi)聯(lián)函數(shù)體來替換。在使用內(nèi)聯(lián)函數(shù)時,應注意如下幾點:
1.在內(nèi)聯(lián)函數(shù)內(nèi)不允許用循環(huán)語句和開關(guān)語句。
如果內(nèi)聯(lián)函數(shù)有這些語句,則編譯將該函數(shù)視同普通函數(shù)那樣產(chǎn)生函數(shù)調(diào)用代碼,遞歸函數(shù)(自己調(diào)用自己的函數(shù))是不能被用來做內(nèi)聯(lián)函數(shù)的。內(nèi)聯(lián)函數(shù)只適合于只有1~5行的小函數(shù)。對一個含有許多語句的大函數(shù),函數(shù)調(diào)用和返回的開銷相對來說微不足道,所以也沒有必要用內(nèi)聯(lián)函數(shù)實現(xiàn)。
2.內(nèi)聯(lián)函數(shù)的定義必須出現(xiàn)在內(nèi)聯(lián)函數(shù)第一次被調(diào)用之前。
3.本欄目講到的類結(jié)構(gòu)中所有在類說明內(nèi)部定義的函數(shù)是內(nèi)聯(lián)函數(shù)。
評論