c++ 虚继承与继承的差异
前面一篇文章,說明了在C++ 虛繼承對基類構(gòu)造函數(shù)調(diào)用順序的影響。經(jīng)過仔細(xì)推敲,發(fā)現(xiàn)沒有徹底說清楚虛繼承與普通繼承之間的關(guān)系。所以用下面的文字再說明一下。
首先,重復(fù)一下虛擬繼承與普通繼承的區(qū)別有:
假設(shè)derived 繼承自base類,那么derived與base是一種“is a”的關(guān)系,即derived類是base類,而反之錯誤;
假設(shè)derived 虛繼承自base類,那么derivd與base是一種“has a”的關(guān)系,即derived類有一個(gè)指向base類的vptr。
因此虛繼承可以認(rèn)為不是一種繼承關(guān)系,而可以認(rèn)為是一種組合的關(guān)系。因?yàn)樘摾^承有著“繼承”兩個(gè)關(guān)鍵字,那么大部分人都認(rèn)為虛繼承與普通繼承的用法沒什么太大的不同。由此用在繼承體系中,這種將虛繼承認(rèn)為是普通繼承的危害更佳大。下面先用一個(gè)例子來說明問題:
[cpp]?view plain?copy
上面是普通繼承實(shí)現(xiàn),在實(shí)際應(yīng)用中,我們可以使用下面的代碼進(jìn)行類型轉(zhuǎn)換:
[cpp]?view plain?copy
?base::base()!
?derived::derived()!
?base::printBase()!
?derived::printDerived()!
而將上面的普通繼承變成虛擬繼承,如下代碼:
[cpp]?view plain?copy
編譯上面的代碼,提示如下:
可以看到不能將基類通過static_cast轉(zhuǎn)換為繼承類。我們知道c++提供的強(qiáng)制轉(zhuǎn)換函數(shù)static_cast對于繼承體系中的類對象的轉(zhuǎn)換一般是可行的。那么這里為什么就不可以了呢?還是需要從虛擬繼承的內(nèi)部實(shí)現(xiàn)來說明問題。
virtual base class的原始模型是在class object中為每一個(gè)有關(guān)聯(lián)的virtual base class加上一個(gè)指針vptr,該指針指向virtual基類表。有的編譯器是在繼承類已存在的virtual table直接擴(kuò)充導(dǎo)入一個(gè)virtual base class table。不管怎么樣由于虛繼承已完全破壞了繼承體系,不能按照平常的繼承體系來進(jìn)行類型轉(zhuǎn)換。
不管怎么樣,虛繼承在類型轉(zhuǎn)換是一定要十分注意。不要輕易使用虛繼承,更不要在虛繼承的基礎(chǔ)上進(jìn)行類型轉(zhuǎn)換,切記切記!
總結(jié)
以上是生活随笔為你收集整理的c++ 虚继承与继承的差异的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: st(state-threads) co
- 下一篇: H.264(MPEG-4 AVC)级别(