一文搞懂C++中繼承的概念與使用
前言
我們都知道面向?qū)ο笳Z言的三大特點(diǎn)是:**封裝,繼承,多態(tài)。**之前在類和對象部分,我們提到了C++中的封裝,那么今天呢,我們來學(xué)習(xí)一下C++中的繼承。
繼承概念及定義
繼承概念
繼承(inheritance)機(jī)制是面向?qū)ο蟪绦蛟O(shè)計(jì)使代碼可以復(fù)用的最重要的手段,它允許程序員在保持原有類特性的基礎(chǔ)上進(jìn)行擴(kuò)展,增加功能,這樣產(chǎn)生新的類,稱派生類。繼承呈現(xiàn)了面向?qū)ο蟪绦蛟O(shè)計(jì)的層次結(jié)構(gòu),體現(xiàn)了由簡單到復(fù)雜的認(rèn)知過程。以前我們接觸的復(fù)用都是函數(shù)復(fù)用,繼承是類設(shè)計(jì)層次的復(fù)用
看概念是一件很讓人疑惑的東西,接下來我就來舉個(gè)例子來看看繼承具體是什么東西
首先我們定義兩個(gè)類,一個(gè)Student類,一個(gè)Teacher類,二者都有年齡和姓名,學(xué)生有學(xué)號,老師有工號。
class Student { private: int _age; //年齡 string _name; //姓名 int _stuid; //學(xué)號 }; class Teacher { private: int _age; //年齡 string _name; //姓名 int _jobid; //工號 };
我們發(fā)現(xiàn)這兩個(gè)類有一些重復(fù)的地方,比如年齡和姓名,這二者是他們的成員,此時(shí)代碼就產(chǎn)生了冗余。那么我們可不可以像個(gè)方法去復(fù)用這兩個(gè)成員呢?繼承此時(shí)就可以發(fā)揮它的重大作用。
我們將他們重復(fù)的地方提取出來,重新定義一個(gè)Person類。而Student類和Teacher類將Person類繼承下來,此時(shí)我們就實(shí)現(xiàn)了代碼的復(fù)用。
class Person { protected: int _age; //年齡 string _name; //姓名 }; class Student : public Person { private: int _stuid; //學(xué)號 }; class Teacher : public Person { private: int _jobid; //工號 };
我們首先分別用Student和Teacher類來創(chuàng)建兩個(gè)對象,來看看對象里面有什么。
int main() { Teacher t; Student s; return 0; }
此時(shí)我們可以看到我們創(chuàng)建的兩個(gè)對象里面都含有從Person類中繼承過來的 age 和 name 兩個(gè)成員。
所以繼承實(shí)際上是一個(gè)代碼的復(fù)用,我們可以借用已完成的類的代碼來完善我們需要創(chuàng)造的新類。
繼承定義
以我們剛剛創(chuàng)建的Student類來舉例:我們看到Person是父類,也稱作基類。Student是子類,也稱作派生類。
繼承方式
我們在類和對象的時(shí)候介紹了三種訪問限定符:public(公有),protected(保護(hù))和private(私有)。訪問限定符限定了我們在類外如何去訪問類中的成員。
在繼承中我們一樣使用這三種限定符來限定子類該如何去訪問父類的的成員,下面有一張表來表示他們的關(guān)系。
類成員\繼承方式 | public繼承 | protected繼承 | private繼承 |
---|---|---|---|
父類的public成員 | 派生類的public成員 | 派生類的protected成員 | 派生類的private成員 |
父類的protected成員 | 派生類的protected成員 | 派生類的protected成員 | 派生類的private成員 |
父類的private成員 | 在派生類中不可見 | 在派生類中不可見 | 在派生類中不可見 |
首先解釋一下在派生類中不可見是什么意思,就如同我們在類外無法直接去修改類中的private成員一樣,我們在子類中也無法直接修改父類的private成員。
如何簡潔的去記這個(gè)表呢?在C++中權(quán)限的關(guān)系:public > protected > private。在繼承的時(shí)候呢,父類成員的權(quán)限取的是:父類成員原本權(quán)限和繼承方式中較小的那個(gè)。
比如父類的A成員原本權(quán)限為public,而子類的繼承方式為private。此時(shí)A成員相對子類來說就為private成員
父類和子類對象賦值轉(zhuǎn)換
子類的對象可以賦值給 父類的對象/父類的指針/父類的應(yīng)用,那么是如何進(jìn)行賦值的呢?形象一點(diǎn)來說就是切片,將子類中父類的部分切割父類。
但我們無法反過來,將父類對象賦值給子類對象。
繼承中的作用域
在繼承體系中父類和子類都有獨(dú)立的作用域,如果子類和父類中有同名的成員,子類成員將屏蔽對父類成員的直接訪問,這種情況叫隱藏,也叫重定義。
下面還是用我們的Person類和Student類來舉個(gè)栗子,我們分別在Person類和Student類中加入一個(gè)print函數(shù),通過打印內(nèi)容來區(qū)分調(diào)用的為哪一print函數(shù)。
class Person { protected: int _age; string _name; public: void print() { cout << "Person"<< endl; } }; class Student : public Person { private: int _stuid; //學(xué)號 public: void print() { cout << "Student" << endl; } };
接下來我們創(chuàng)建一個(gè)對象然后來試一下結(jié)果。
int main() { Student s; s.print(); return 0; }
我們可以看到我們調(diào)用的為Student中的print函數(shù)。此時(shí)子類的print函數(shù)已經(jīng)對父類的print函數(shù)進(jìn)行了重定義。重定義不代表子類無法去調(diào)用父類的同名函數(shù),只是不那么直接而已。使用下面這種方法我們就可以調(diào)用父類中的同名函數(shù)。
int main() { Student s; s.Person::print(); return 0; }
通過指定類域,我們就可以去調(diào)用父類的print函數(shù)。但在實(shí)際中最好不要去定義同名函數(shù)以免帶來問題。
派生類的默認(rèn)成員函數(shù)
首先我們來回顧一下有哪幾個(gè)默認(rèn)成員函數(shù)。
那么在子類中,這些默認(rèn)成員函數(shù)是怎么生成的呢?
1.子類的構(gòu)造函數(shù)必須調(diào)用父類的構(gòu)造函數(shù)初始化父類的那一部分成員。如果父類沒有默認(rèn)的構(gòu)造函數(shù),則必須在派生類構(gòu)造函數(shù)的初始化列表中顯式調(diào)用。還是用我們的Person類和Student類舉例。
情況一:有默認(rèn)構(gòu)造函數(shù)
class Person { protected: int _age; string _name; public: Person() { cout << "Person" << endl; //調(diào)用就打印 } }; class Student : public Person { private: int _stuid; //學(xué)號 public: Student() { cout << "Student" << endl; //調(diào)用就打印 } }; int main() { Student s; return 0; }
情況二:無默認(rèn)構(gòu)造函數(shù)
class Person { protected: int _age; string _name; public: Person(int age, string name) { cout << "Person" << endl; } }; class Student : public Person { private: int _stuid; //學(xué)號 public: Student() : Person(19, "wanku") //無默認(rèn)構(gòu)造,此時(shí)我們需要在初始化列表中初始化 { cout << "Student" << endl; } }; int main() { Student s; return 0; } int main() { Student s; return 0; }
2.子類的拷貝構(gòu)造函數(shù)必須調(diào)用父類的拷貝構(gòu)造完成父類的拷貝初始化化。
class Person { protected: int _age; string _name; public: Person(int age = 10, string name = "wanku") { cout << "Person" << endl; } Person(const Person &p) : _age(p._age), _name(p._name) {} }; class Student : public Person { private: int _stuid; //學(xué)號 public: Student() { cout << "Student" << endl; } Student(const Student &s) : Person(s) /*顯示調(diào)用父類的拷貝構(gòu)造*/, _stuid(s._stuid) {} };
有些朋友可能會疑惑,在Person類中的拷貝構(gòu)造函數(shù)參數(shù)明明是Person類,為什么我們的Student類可以傳過去呢?那是因?yàn)槲覀儎倓傊v的切片原理,當(dāng)我們把子類對象傳過去時(shí),編譯器會進(jìn)行切分,然后再傳給父類。
3.派生類的operator=必須要調(diào)用基類的operator=完成基類的復(fù)制。(原理和拷貝構(gòu)造大體相似,值得注意的是:當(dāng)我們在子類直接想去調(diào)用父類的operator= 時(shí),會發(fā)生重定義,使用時(shí)記得加上父類的作用域)
4.在繼承中一個(gè)對象的歷程如下:父類的構(gòu)造函數(shù) –> 子類的構(gòu)造函數(shù) –> 子類的析構(gòu)函數(shù) –> 父類的析構(gòu)函數(shù)。這個(gè)過程相當(dāng)于把這些行為存在一個(gè)棧中,然后再把行為從棧中拿出來一般
派生類的友元與靜態(tài)成員
父類的友元不是子類的友元。(你爸爸的朋友不一定是你的朋友)
父類中有一個(gè)靜態(tài)成員,那么子類和父類共用一個(gè)靜態(tài)成員。(靜態(tài)成員并不存在對象中,只開辟一個(gè)空間,所以只能共用一個(gè))
繼承關(guān)系
單繼承
一個(gè)子類只有一個(gè)直接父類。
多繼承
一個(gè)子類有兩個(gè)及以上的父類
菱形繼承
多繼承的一種特殊情況
以上就是一文搞懂C++中繼承的概念與使用的詳細(xì)內(nèi)容,更多關(guān)于C++繼承的資料請關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
C語言 解決不用+、-、×、÷數(shù)字運(yùn)算符做加法的實(shí)現(xiàn)方法
本篇文章是對在C語言中解決不用+、-、×、÷數(shù)字運(yùn)算符做加法的方法進(jìn)行了詳細(xì)的分析介紹,需要的朋友參考下2013-05-05詳解C++類的成員函數(shù)做友元產(chǎn)生的循環(huán)依賴問題
這篇文章主要為大家詳細(xì)介紹了C++類的成員函數(shù)做友元產(chǎn)生的循環(huán)依賴問題,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下,希望能夠給你帶來幫助2022-03-03C++對象內(nèi)存分布詳解(包括字節(jié)對齊和虛函數(shù)表)
下面小編就為大家?guī)硪黄狢++對象內(nèi)存分布詳解(包括字節(jié)對齊和虛函數(shù)表)。小編覺得挺不錯的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧2016-12-12C++實(shí)現(xiàn)經(jīng)典24點(diǎn)紙牌益智游戲
這篇文章主要介紹了C++實(shí)現(xiàn)經(jīng)典24點(diǎn)紙牌益智游戲,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2020-03-03c++將數(shù)組名作為函數(shù)參數(shù)對數(shù)組元素進(jìn)行相應(yīng)的運(yùn)算
這篇文章主要介紹了c++將數(shù)組名作為函數(shù)參數(shù)對數(shù)組元素進(jìn)行相應(yīng)的運(yùn)算,需要的朋友可以參考下2014-05-05Qt?Creator配置opencv環(huán)境的全過程記錄
最近在PC端QT下配置opencv,想著以后應(yīng)該會用到,索性記錄下,這篇文章主要給大家介紹了關(guān)于Qt?Creator配置opencv環(huán)境的相關(guān)資料,需要的朋友可以參考下2022-05-05