多重繼承
物件導向程式設計中的多重繼承(英語:multiple inheritance,縮寫:MI)指的是一個類別可以同時從多於一個父類繼承行為與特徵的功能。與單一繼承相對,單一繼承指一個類別只可以繼承自一個父類。
爭議
多重繼承可以導致某些令人混淆的情況,所以關於它的好處與風險之間孰輕孰重常常受人爭論。Java使用了一個折衷的辦法:Java允許一個類別繼承自多於一個父介面(可以指定某一個類別,它繼承了所有父類別的類型,並必須擁有所有父類別介面的外部可見方法的具體實現,並允許編譯器強制以上要求),但只可以從一個父類別繼承實現(方法與數據)。微軟的.NET程式語言,例如C#和Visual Basic .NET也使用了這種介面的做法。
物件導向的程式設計中,繼承描述了兩種類型或兩個類的物件,其中一種是另外一種的「子類型」或「子類別」。子類別繼承了父類別的特徵,允許分享功能。例如,可以創造一個「哺乳類動物」類別,擁有進食、繁殖等的功能;然後定義一個子類型「貓」,它可以從父類別繼承上述功能,不需重新編寫程式,同時增加屬於自己的新功能,例如「追趕老鼠」。
然而,如果想同時自多於一個結構繼承,例如容許「貓」繼承「哺乳類動物」之餘,同時繼承「卡通角色」和「寵物」,缺乏多重繼承往往會導致十分笨拙的混合繼承,或迫使同一個功能在多於一個地方被重寫。(這帶來了維護上的問題)
多年以來,多重繼承都是一個敏感的話題,反對者指它增加了程式的複雜性與含糊性,例如在鑽石問題(或稱菱型缺陷)中。Loki函式庫針對多重繼承進行改良,以TypeList(二叉樹結構)避免這個問題。
各種程式語言有不同的方式處理上述問題。例如Eiffel容許子類型透過重新命名,或提前為他們確定選擇規則,來適應(adapt)它繼承得來的功能。Java允許物件從多個介面繼承,但僅允許一個實現繼承。REALbasic與它相似,並增加了一個不需使用繼承來「擴展」一個類別的功能。Perl使用一種有序列表式的繼承機制:搜尋方法時,它會先搜尋當前類別的方法,然後使用深度優先搜尋來順序尋找各個繼承類別及其父類別。CLOS允許程式設計者完全控制方法的組合。如果這還不足夠,元對象協定給程式設計者一種手段去修改繼承,方法調度,類別特例化,及其它內部的機制,而不影響系統的穩定性。
C++與多繼承
C++支援多重繼承,允許對現實世界進行更直接的建模,Borland C++的OWL Framework大量使用多重繼承來描述視窗的關係。微軟的MFC僅使用單一繼承描述視窗,ATL使用多重繼承實現COM/ActiveX,WTL則使用多重繼承實現視窗。
多重繼承與被覆蓋的虛擬函式
對於最左基礎類別,虛擬函式的覆蓋與單繼承情形一致。
對於非最左的基礎類別,虛擬函式仍然可能會被衍生類別的成員函數覆蓋。
成員函數中this指標調整
一個類的非靜態成員函數,一般需要使用類對象的this指標來訪問類數據成員。程式載入到主記憶體後,成員函數代碼佔據了一塊主記憶體空間。成員函數並不知道自身是作為一個單獨的(或最衍生)類的直接成員函數,還是作為一個被衍生的基礎類別的成員函數而存在。實際上在主記憶體空間的非靜態成員函數,可能會同時是單獨的(或最衍生)類的直接成員函數與被衍生的基礎類別的成員函數。非靜態成員函數也僅知道聲明了該函數的類的數據成員的空間分佈,不可能知道以該類為基礎類別的衍生類別的數據成員的空間分佈。因此呼叫非靜態成員函數時,呼叫者有責任傳給成員函數正確的this指標,即令this指標指向聲明了該成員函數的類的對象開始地址。
對於單繼承,衍生類別與基礎類別的對象開始地址是一樣的,因此呼叫非靜態成員函數不需要調整this指標。對於多繼承,呼叫不是最左基礎類別的非靜態成員函數時,呼叫者必須先調整this指標。這又分為兩種情形:
一是非虛擬函式,在函數呼叫現場直接調整this的值。這是編譯器根據多重繼承的衍生類別的實例對象或指標在編譯時就能確定的。例如:
struct base1{
int v1;
void foo1(int){}
}
struct base2{
int v2;
void foo2(int){}
}
struct derive: base1,base2{
};
derive d;
int main()
{
derive *p=&d;
d.foo2(101);
/* 上述调用语句编译后为:
push 65h ;参数101压栈
lea ecx,offset d+4 ;根据thiscall调用协议,ecx保存了this的值
call base2::foo2 (1181145h)
*/
p->foo2(102);
}
二是虛擬函式情形。因為虛擬函式的開始地址必須存放在虛表條目中,所以多重繼承的衍生類別對非最左基礎類別的被覆蓋(override)的虛擬函式,在該衍生類別的相應的虛表條目中填寫的是一個樁(thunk)地址。該樁通常只有兩條機器指令,首先是調整this值(即修改ecx暫存器),然後是呼叫指令(call)。
參考文獻
- Andrei Alexandrescu. Modern C++ Design
外部連結
- (英文) Article by Jonathan Lurie of Builder.Com on instances in .NET languages: https://web.archive.org/web/20060112044813/http://builder.com.com/5100-6373-5030734.html