一文搞懂C++中繼承的概念與使用
前言
我們都知道面向?qū)ο笳Z言的三大特點(diǎn)是:**封裝,繼承,多態(tài)。**之前在類和對象部分,我們提到了C++中的封裝,那么今天呢,我們來學(xué)習(xí)一下C++中的繼承。
繼承概念及定義
繼承概念
繼承(inheritance)機(jī)制是面向?qū)ο蟪绦蛟O(shè)計使代碼可以復(fù)用的最重要的手段,它允許程序員在保持原有類特性的基礎(chǔ)上進(jìn)行擴(kuò)展,增加功能,這樣產(chǎn)生新的類,稱派生類。繼承呈現(xiàn)了面向?qū)ο蟪绦蛟O(shè)計的層次結(jié)構(gòu),體現(xiàn)了由簡單到復(fù)雜的認(rèn)知過程。以前我們接觸的復(fù)用都是函數(shù)復(fù)用,繼承是類設(shè)計層次的復(fù)用
看概念是一件很讓人疑惑的東西,接下來我就來舉個例子來看看繼承具體是什么東西
首先我們定義兩個類,一個Student類,一個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)這兩個類有一些重復(fù)的地方,比如年齡和姓名,這二者是他們的成員,此時代碼就產(chǎn)生了冗余。那么我們可不可以像個方法去復(fù)用這兩個成員呢?繼承此時就可以發(fā)揮它的重大作用。
我們將他們重復(fù)的地方提取出來,重新定義一個Person類。而Student類和Teacher類將Person類繼承下來,此時我們就實現(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)建兩個對象,來看看對象里面有什么。
int main()
{
Teacher t;
Student s;
return 0;
}
此時我們可以看到我們創(chuàng)建的兩個對象里面都含有從Person類中繼承過來的 age 和 name 兩個成員。
所以繼承實際上是一個代碼的復(fù)用,我們可以借用已完成的類的代碼來完善我們需要創(chuàng)造的新類。
繼承定義
以我們剛剛創(chuàng)建的Student類來舉例:我們看到Person是父類,也稱作基類。Student是子類,也稱作派生類。

繼承方式
我們在類和對象的時候介紹了三種訪問限定符:public(公有),protected(保護(hù))和private(私有)。訪問限定符限定了我們在類外如何去訪問類中的成員。
在繼承中我們一樣使用這三種限定符來限定子類該如何去訪問父類的的成員,下面有一張表來表示他們的關(guān)系。
| 類成員\繼承方式 | public繼承 | protected繼承 | private繼承 |
|---|---|---|---|
| 父類的public成員 | 派生類的public成員 | 派生類的protected成員 | 派生類的private成員 |
| 父類的protected成員 | 派生類的protected成員 | 派生類的protected成員 | 派生類的private成員 |
| 父類的private成員 | 在派生類中不可見 | 在派生類中不可見 | 在派生類中不可見 |
首先解釋一下在派生類中不可見是什么意思,就如同我們在類外無法直接去修改類中的private成員一樣,我們在子類中也無法直接修改父類的private成員。
如何簡潔的去記這個表呢?在C++中權(quán)限的關(guān)系:public > protected > private。在繼承的時候呢,父類成員的權(quán)限取的是:父類成員原本權(quán)限和繼承方式中較小的那個。
比如父類的A成員原本權(quán)限為public,而子類的繼承方式為private。此時A成員相對子類來說就為private成員
父類和子類對象賦值轉(zhuǎn)換
子類的對象可以賦值給 父類的對象/父類的指針/父類的應(yīng)用,那么是如何進(jìn)行賦值的呢?形象一點(diǎn)來說就是切片,將子類中父類的部分切割父類。

但我們無法反過來,將父類對象賦值給子類對象。
繼承中的作用域
在繼承體系中父類和子類都有獨(dú)立的作用域,如果子類和父類中有同名的成員,子類成員將屏蔽對父類成員的直接訪問,這種情況叫隱藏,也叫重定義。
下面還是用我們的Person類和Student類來舉個栗子,我們分別在Person類和Student類中加入一個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)建一個對象然后來試一下結(jié)果。
int main()
{
Student s;
s.print();
return 0;
}
我們可以看到我們調(diào)用的為Student中的print函數(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ù)以免帶來問題。
派生類的默認(rèn)成員函數(shù)
首先我們來回顧一下有哪幾個默認(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)造,此時我們需要在初始化列表中初始化
{
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類可以傳過去呢?那是因為我們剛剛講的切片原理,當(dāng)我們把子類對象傳過去時,編譯器會進(jìn)行切分,然后再傳給父類。
3.派生類的operator=必須要調(diào)用基類的operator=完成基類的復(fù)制。(原理和拷貝構(gòu)造大體相似,值得注意的是:當(dāng)我們在子類直接想去調(diào)用父類的operator= 時,會發(fā)生重定義,使用時記得加上父類的作用域)
4.在繼承中一個對象的歷程如下:父類的構(gòu)造函數(shù) –> 子類的構(gòu)造函數(shù) –> 子類的析構(gòu)函數(shù) –> 父類的析構(gòu)函數(shù)。這個過程相當(dāng)于把這些行為存在一個棧中,然后再把行為從棧中拿出來一般
派生類的友元與靜態(tài)成員
父類的友元不是子類的友元。(你爸爸的朋友不一定是你的朋友)
父類中有一個靜態(tài)成員,那么子類和父類共用一個靜態(tài)成員。(靜態(tài)成員并不存在對象中,只開辟一個空間,所以只能共用一個)
繼承關(guān)系
單繼承
一個子類只有一個直接父類。

多繼承
一個子類有兩個及以上的父類

菱形繼承
多繼承的一種特殊情況

以上就是一文搞懂C++中繼承的概念與使用的詳細(xì)內(nèi)容,更多關(guān)于C++繼承的資料請關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
C++ 標(biāo)準(zhǔn)模板庫 STL 順序容器詳解
這篇文章主要介紹了C++ 標(biāo)準(zhǔn)模板庫 STL 順序容器詳解,本文給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下2021-05-05
C語言設(shè)置和取得socket狀態(tài)的相關(guān)函數(shù)用法
這篇文章主要介紹了C語言設(shè)置和取得socket狀態(tài)的相關(guān)函數(shù)用法,分別是setsockopt()函數(shù)和getsockopt()函數(shù)的使用介紹,需要的朋友可以參考下2015-09-09
C/C++實現(xiàn)枚舉網(wǎng)上鄰居信息的示例詳解
在Windows系統(tǒng)中,通過網(wǎng)絡(luò)鄰居可以方便地查看本地網(wǎng)絡(luò)中的共享資源和計算機(jī),本文將介紹一個簡單的C++程序,使用Windows API枚舉網(wǎng)絡(luò)鄰居信息,并獲取對端名稱、本機(jī)名稱、主機(jī)名稱以及主機(jī)IP等信息,文中通過代碼示例給大家講解非詳細(xì),需要的朋友可以參考下2023-12-12

