網(wǎng)站首頁(yè) 編程語(yǔ)言 正文
封裝,繼承,多態(tài)。這是C++語(yǔ)言的三大特性,而每次在談到繼承時(shí)我們不可避免的要談到一個(gè)很重要的問(wèn)題——菱形繼承。
派生類繼承父類,同時(shí)也會(huì)繼承父類中的所有成員副本,但如果在繼承時(shí)一個(gè)基類同時(shí)被兩個(gè)子類繼承,然后一個(gè)新類又分別由上面的兩個(gè)子類派生出來(lái)。這樣從某種程度來(lái)說(shuō)就形成了C++中的菱形繼承,也可以叫做鉆石繼承,具體的繼承形式如下圖所示:
在上面的類圖說(shuō),Left和Right分別派生子Top,但是Bottom又分別繼承了Left和Right。繼承關(guān)系也可以畫(huà)成下面的方式,這樣就可以更好的理解設(shè)計(jì)中存在的問(wèn)題。
該類圖很明確的展示了類設(shè)計(jì)中的不足之處,在試圖將指向Bottom對(duì)象的指針轉(zhuǎn)換成指向Top的指針時(shí),有兩個(gè)Top對(duì)象可供選擇,但是編譯器卻明顯沒(méi)有那么智能,從而導(dǎo)致了轉(zhuǎn)換過(guò)程中的二義性;同理,Bottom對(duì)象也不能直接調(diào)用Top中定義的方法,如果要使用需要提供一個(gè)Top子對(duì)象,但是從類圖可知存在兩個(gè)Top對(duì)象。
上面的類對(duì)應(yīng)的代碼為:
class Top{ public: int _x; public: Top(int x):_x(x){}; }; class Left:public Top{ public: int _y; public: Left(int x,int y):Top(x),_y(y){} }; class Right:public Top{ public: int _z; public: Right(int x,int z):Top(x),_z(z){} }; class Bottom:public Left,public Right{ public: int _w; public: Bottom(int x,int y,int z,int w):Left(x,z),Right(y,z),_w(w){}; };
下面實(shí)現(xiàn)該類的測(cè)試程序,如下所示:
int main() { Bottom bf(1,2,3,4); cout<<sizeof(bf)<<endl; return 0; }
運(yùn)行結(jié)果為:20,在打印基類中的成員時(shí)編譯器也會(huì)報(bào)以下錯(cuò)誤:
既然在上面的類的設(shè)計(jì)中存在問(wèn)題,在實(shí)際編程時(shí)如何避免這個(gè)問(wèn)題呢?
答案是:虛基類。
虛基類給在確實(shí)需要使用菱形繼承的地方提供了一個(gè)很好的解決方法,通過(guò)子類共享一個(gè)基類對(duì)象避免基類對(duì)象的二義性問(wèn)題。
上面的代碼修改后代碼如下:
using namespace std; class Top{ public: int _x; public: Top(int x):_x(x){}; virtual ~Top(){}; }; class Left:virtual public Top{ public: int _y; public: Left(int x,int y):Top(x),_y(y){} }; class Right:virtual public Top{ public: int _z; public: Right(int x,int z):Top(x),_z(z){} }; class Bottom:public Left,public Right{ public: int _w; public: Bottom(int x,int y,int z,int w):Top(x),Left(x,y),Right(x,z),_w(w){}; };
在main函數(shù)中繼續(xù)測(cè)試上述類,則可以正常輸出,代碼如下:
int main() { Bottom bf(1,2,3,4); cout<<bf._x<<","<<bf._y<<","<<bf._z<<","<<bf._w<<endl; return 0; }
運(yùn)行結(jié)果為:
從上面的示例可以看出,在使用多進(jìn)程時(shí)如果不對(duì)類進(jìn)行提前規(guī)劃,將可能產(chǎn)生菱形繼承這種場(chǎng)景,給實(shí)際的編程帶來(lái)不便。因此在實(shí)際編碼時(shí),我建議盡量減少多繼承的方式更多地使用嵌套類的方式。
總結(jié)
原文鏈接:https://mp.weixin.qq.com/s/OBSTK3kvjvqEpbmj8vXzpQ
相關(guān)推薦
- 2022-09-21 Oracle數(shù)據(jù)庫(kù)對(duì)象的使用詳解_oracle
- 2022-04-06 Python數(shù)據(jù)結(jié)構(gòu)之循環(huán)鏈表詳解_python
- 2022-11-03 一文詳解C++子類函數(shù)為什么不能重載父類函數(shù)_C 語(yǔ)言
- 2023-07-13 react父子組件,任意組件傳值
- 2022-06-02 基于python的MD5腳本開(kāi)發(fā)思路_python
- 2022-06-17 go語(yǔ)言beego框架jwt身份認(rèn)證實(shí)現(xiàn)示例_Golang
- 2022-01-04 變量提升,函數(shù)提升及其優(yōu)先級(jí)關(guān)系
- 2022-04-09 cas5 編譯安裝依賴時(shí)提示: Failure to find net.shibboleth.too
- 最近更新
-
- window11 系統(tǒng)安裝 yarn
- 超詳細(xì)win安裝深度學(xué)習(xí)環(huán)境2025年最新版(
- Linux 中運(yùn)行的top命令 怎么退出?
- MySQL 中decimal 的用法? 存儲(chǔ)小
- get 、set 、toString 方法的使
- @Resource和 @Autowired注解
- Java基礎(chǔ)操作-- 運(yùn)算符,流程控制 Flo
- 1. Int 和Integer 的區(qū)別,Jav
- spring @retryable不生效的一種
- Spring Security之認(rèn)證信息的處理
- Spring Security之認(rèn)證過(guò)濾器
- Spring Security概述快速入門
- Spring Security之配置體系
- 【SpringBoot】SpringCache
- Spring Security之基于方法配置權(quán)
- redisson分布式鎖中waittime的設(shè)
- maven:解決release錯(cuò)誤:Artif
- restTemplate使用總結(jié)
- Spring Security之安全異常處理
- MybatisPlus優(yōu)雅實(shí)現(xiàn)加密?
- Spring ioc容器與Bean的生命周期。
- 【探索SpringCloud】服務(wù)發(fā)現(xiàn)-Nac
- Spring Security之基于HttpR
- Redis 底層數(shù)據(jù)結(jié)構(gòu)-簡(jiǎn)單動(dòng)態(tài)字符串(SD
- arthas操作spring被代理目標(biāo)對(duì)象命令
- Spring中的單例模式應(yīng)用詳解
- 聊聊消息隊(duì)列,發(fā)送消息的4種方式
- bootspring第三方資源配置管理
- GIT同步修改后的遠(yuǎn)程分支