亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 學院 > 開發設計 > 正文

C++--第五課-繼承

2019-11-14 08:48:30
字體:
來源:轉載
供稿:網友

繼承中的幾種關系

1. is a的關系

比如說基類狗,派生了肉狗和寵物狗,我們就可以說肉狗是狗,也可以說寵物狗是狗,這就是is a的關系。但是需要注意的是,并非所有的繼承都是is a的關系。

2. has a的關系

繼承中的權限繼承

class Base{public: int public_i_; // 外部可訪問PRotected: int protected_i_; // 內部可訪問,外部無法訪問, 可繼承,子類可訪問private: int private_i_; // 內部可訪問,外部不可訪問, 不可繼承,子類不可訪問};class A : public Base // public繼承所有的權限都沒有發生改變{ void SetI() { protected_i_ = 100; }};class B : protected Base // protected 繼承會將基類中public的權限改為protected權限{ void SetI() { protected_i_ = 100; }};class C : private Base // 將基類中的protected和public權限都改為private權限{ void SetI() { protected_i_ = 100; }};

基類中有默認構造函數的繼承中的構造和析構順序

#include <iostream>class Base{public: Base() { std::cout << "Base::Base()" << std::endl; } ~Base() { std::cout << "Base::~Base()" << std::endl; } int GetNum() { return num_; }private: int num_;};class A : public Base{public: A() { std::cout << "A::A()" << std::endl; } ~A() { std::cout << "A::~A()" << std::endl; }};int main(){ A demo; demo.GetNum(); return 0;}

這里寫圖片描述

基類中沒有默認構造函數的繼承中的構造和析構順序

#include <iostream>class Base{public: Base(int num) : num_(num) { std::cout << "Base::Base()" << std::endl; } ~Base() { std::cout << "Base::~Base()" << std::endl; } int GetNum() { return num_; }private: int num_;};class A : public Base{public: A() : Base(10) // 此時會默認的調用基類的默認構造函數,如果基類中沒有默認的構造函數時, // 那么必須在派生類中的初始化列表中顯示的來構造 { std::cout << "A::A()" << std::endl; } ~A() { std::cout << "A::~A()" << std::endl; }};int main(){ A demo; demo.GetNum(); return 0;}

繼承中的函數,派生類中沒有實現基類的函數方法

#include <iostream>class Base{public: Base(int num) : num_(num) { std::cout << "Base::Base()" << std::endl; } ~Base() { std::cout << "Base::~Base()" << std::endl; } int GetNum() const { return num_; }private: int num_;};class A : public Base{public: A(int num) : Base(0), num_(num) // 此時會默認的調用基類的默認構造函數,如果基類中沒有默認的構造函數時, // 那么必須在派生類中的初始化列表中顯示的來構造 { std::cout << "A::A()" << std::endl; } ~A() { std::cout << "A::~A()" << std::endl; } //int GetNum() const //{ // return num_; //}private: int num_;};int main(){ A demo(100); demo.GetNum(); std::cout << demo.GetNum() << std::endl; return 0;}

結果是 這里寫圖片描述

另一種情況是,派生類中實現了基類的函數方法

#include <iostream>class Base{public: Base(int num) : num_(num) { std::cout << "Base::Base()" << std::endl; } ~Base() { std::cout << "Base::~Base()" << std::endl; } int GetNum() const { return num_; }private: int num_;};class A : public Base{public: A(int num) : Base(0), num_(num) // 此時會默認的調用基類的默認構造函數,如果基類中沒有默認的構造函數時, // 那么必須在派生類中的初始化列表中顯示的來構造 { std::cout << "A::A()" << std::endl; } ~A() { std::cout << "A::~A()" << std::endl; } int GetNum() const { return num_; }private: int num_;};int main(){ A demo(100); demo.GetNum(); std::cout << demo.GetNum() << std::endl; return 0;}

輸出結果是 這里寫圖片描述

基類指針指向派生類對象的函數調用情況

#include <iostream>class Base{public: Base(int num) : num_(num) { std::cout << "Base::Base()" << std::endl; } ~Base() { std::cout << "Base::~Base()" << std::endl; } int GetNum() const { std::cout << "Base::GetNum()" << std::endl; return num_; }private: int num_;};class A : public Base{public: A(int num) : Base(0), num_(num) // 此時會默認的調用基類的默認構造函數,如果基類中沒有默認的構造函數時, // 那么必須在派生類中的初始化列表中顯示的來構造 { std::cout << "A::A()" << std::endl; } ~A() { std::cout << "A::~A()" << std::endl; } int GetNum() const // 這個函數被重寫了,也就是把基類中的方法給覆蓋了,基類中的方法就不存在了 { std::cout << "A::GetNum()" << std::endl; return num_; }private: int num_;};int main(){ Base base(100); A a(200); Base *pBase = &a; // 因為 A is Base // 寵物狗 Is 狗 對 狗 泛指 寵物狗 // 狗 Is 寵物狗 錯 std::cout << pBase->GetNum() << std::endl; // 調用的是 A 中的 Base中的GetNum方法,但是這樣很顯然不是我們想要的結果 // 這種用基類的指針指向派生類的對象的情況很多 return 0;}

輸出結果是 這里寫圖片描述

使用虛函數在繼承中函數調用的好處

#include <iostream>class Base{public: Base(int num) : num_(num) { std::cout << "Base::Base()" << std::endl; } ~Base() { std::cout << "Base::~Base()" << std::endl; } virtual int GetNum() const { std::cout << "Base::GetNum()" << std::endl; return num_; }private: int num_;};class A : public Base{public: A(int num) : Base(0), num_(num) // 此時會默認的調用基類的默認構造函數,如果基類中沒有默認的構造函數時, // 那么必須在派生類中的初始化列表中顯示的來構造 { std::cout << "A::A()" << std::endl; } ~A() { std::cout << "A::~A()" << std::endl; } int GetNum() const // 這個函數被重寫了,也就是把基類中的方法給覆蓋了,基類中的方法就不存在了 { std::cout << "A::GetNum()" << std::endl; return num_; }private: int num_;};int main(){ Base base(100); A a(200); Base *pBase = &a; // 因為 A is Base // 寵物狗 Is 狗 對 狗 泛指 寵物狗 // 狗 Is 寵物狗 錯 std::cout << pBase->GetNum() << std::endl; // 調用的是 A 中的 Base中的GetNum方法,但是這樣很顯然不是我們想要的結果 // ,使用虛函數就可以滿足我們的需求,我們在基類中的GetNum前面加上virtual關鍵字 // 這種用基類的指針指向派生類的對象的情況很多 return 0;}

結果是 這里寫圖片描述 為什么會這樣呢?因為使用了虛函數,虛函數用來指明派生類中需要調用的函數,也就是說會在派生類中優先尋找要調用的函數,而不是在基類中尋找這個函數。

虛函數的其它例子

派生類繼承并操作了基類中的成員變量

繼承了基類中的str_變量,并對其進行了操作,導致內存泄漏,并且在析構的時候重復析構。

#define _CRT_SECURE_NO_WARNINGS#include <iostream>#include <cstring>class String{public: String(const char *str = "") { unsigned int len = strlen(str); str_ = new char[len + sizeof(char)]; strcpy(str_, str); } ~String() { delete[] str_; } String &Operator+=(const String &other) { unsigned int len = strlen(str_) + strlen(other.str_); char *temp = new char[len + sizeof(char)]; strcpy(temp, str_); strcat(temp, other.str_); delete[] str_; str_ = temp; return *this; } String operator+(const String &other) { String demo; demo += other; return demo; }protected: char *str_;};class MyString : public String{public: MyString(const char *str = "PoEdu") { unsigned int len = strlen(str); str_ = new char[len + sizeof(char)]; strcpy(str_, str); } MyString operator+(const MyString &other) { } ~MyString() { delete[] str_; str_ = nullptr; }};int main(){ MyString str; //MyString other("Hello"); return 0;}
調用基類的構造函數來操作基類的成員變量,而不是在派生類中直接操作基類只能怪的成員變量
#define _CRT_SECURE_NO_WARNINGS#include <iostream>#include <cstring>class String{public: String(const char *str = "") { unsigned int len = strlen(str); str_ = new char[len + sizeof(char)]; strcpy(str_, str); } ~String() { delete[] str_; } String &operator+=(const String &other) { unsigned int len = strlen(str_) + strlen(other.str_); char *temp = new char[len + sizeof(char)]; strcpy(temp, str_); strcat(temp, other.str_); delete[] str_; str_ = temp; return *this; } String operator+(const String &other) { String demo; demo += other; return demo; }protected: char *str_;};class MyString : public String{public: MyString(const char *str = "PoEdu") : String(str) { /*unsigned int len = strlen(str); str_ = new char[len + sizeof(char)]; strcpy(str_, str);*/ } MyString operator+(const MyString &other) { } ~MyString() { delete[] str_; str_ = nullptr; }};int main(){ MyString str; //MyString other("Hello"); return 0;}

這樣雖然保證了只對基類中的成員變量操作一次,但是在析構的時候還是會重復析構。

虛析構函數的使用情形

沒有虛析構函數的析構情形

#define _CRT_SECURE_NO_WARNINGS#include <iostream>#include <cstring>class A{public: ~A() { std::cout << "~A()" << std::endl; }};class B : public A{public: ~B() { std::cout << "~B(()" << std::endl; }};int main(){ A *pA = new B(); delete pA; return 0;}

運行結果如下圖所示 這里寫圖片描述 這樣只是調用了基類中的析構函數,并沒有調用派生類中的析構函數

有虛析構函數的析構情形

#define _CRT_SECURE_NO_WARNINGS#include <iostream>#include <cstring>class A{public: virtual ~A() { std::cout << "~A()" << std::endl; }};class B : public A{public: ~B() { std::cout << "~B(()" << std::endl; }};int main(){ A *pA = new B(); delete pA; return 0;}

運行結果如下: 這里寫圖片描述 這樣就會把派生類中的析構函數也調用了。

正確的做法是只需要管理好本類中的成員變量即可,無需管理繼承下來的成員變量(其實我覺得在實際的應用中,成員變量應該都是private的,是不需要被子類繼承的,只有函數方法才需要被繼承下來)
#define _CRT_SECURE_NO_WARNINGS#include <iostream>#include <cstring>class String{public: String(const char *str = "") { unsigned int len = strlen(str); str_ = new char[len + sizeof(char)]; strcpy(str_, str); } ~String() { delete[] str_; } String &operator+=(const String &other) { unsigned int len = strlen(str_) + strlen(other.str_); char *temp = new char[len + sizeof(char)]; strcpy(temp, str_); strcat(temp, other.str_); delete[] str_; str_ = temp; return *this; } String operator+(const String &other) { String demo; demo += other; return demo; }protected: char *str_;};class MyString : public String{public: MyString(const char *str = "PoEdu") : String(str) { length_ = new int(0); /*unsigned int len = strlen(str); str_ = new char[len + sizeof(char)]; strcpy(str_, str);*/ } MyString operator+(const MyString &other) { } ~MyString() { //delete[] str_; //str_ = nullptr; // 正確的方法是不要去管理基類中的成員變量,只需要管理本類中的成員變量即可 delete length_; // 只需要管理本類中的length變量就可以了 }private: int *length_;};int main(){ //MyString str; //MyString other("Hello"); return 0;}

這樣就保證了程序的正確運行。

虛函數的調用

#define _CRT_SECURE_NO_WARNINGS#include <iostream>#include <cstring>class String{public: String(const char *str = "") { unsigned int len = strlen(str); str_ = new char[len + sizeof(char)]; strcpy(str_, str); } String(const String &other) { unsigned int len = strlen(other.str_); str_ = new char[len + sizeof(char)]; strcpy(str_, other.str_); } ~String() { delete[] str_; } String &operator+=(const String &other) { unsigned int len = strlen(str_) + strlen(other.str_); char *temp = new char[len + sizeof(char)]; strcpy(temp, str_); strcat(temp, other.str_); delete[] str_; str_ = temp; return *this; } String operator+(const String &other) { String demo(*this); demo += other; return demo; } friend std::ostream &operator<<(std::ostream &os, const String &other) { os << other.str_; return os; }protected: char *str_;};class MyString : public String{public: MyString(const char *str = "PoEdu") : String(str) { } MyString operator+(const MyString &other) { MyString temp(*this); temp += other; // 因為從基類中繼承了一個 += 操作符 temp += "---------PoEdu"; return temp; // 此時在返回的時候會調用拷貝構造函數,如果沒有實現拷貝構造函數, // 那么就會出現淺拷貝的情況,導致程序運行錯誤,所以我們必須手動實現拷貝構造函數 }};int main(){ MyString str("I Love "); String *pString = &str; std::cout << str + ("Mark") << std::endl; std::cout << *pString + ("Mark") << std::endl; return 0;}

運行結果 這里寫圖片描述

繼承中的虛析構函數

基類中的析構函數不是虛析構函數,而是普通函數的情況:

#define _CRT_SECURE_NO_WARNINGS#include <iostream>#include <cstring>class A{public: ~A() { std::cout << "~A()" << std::endl; }};class B : public A{public: B() : A() { demo_ = new int(0); } ~B() { std::cout << "~B()" << std::endl; delete demo_; }private: int *demo_;};int main(){ A * pA = new B(); delete pA; return 0;}

運行結果是: 這里寫圖片描述 從結果上可以看出,在delete的時候,只是調用了基類中的析構函數,并沒有調用派生類中的析構函數,那么,如何能讓派生類中的析構函數也被調用呢?請看下面的情況?。?! 基類中的析構函數是虛析構函數的情況:

#define _CRT_SECURE_NO_WARNINGS#include <iostream>#include <cstring>class A{public: virtual ~A() { std::cout << "~A()" << std::endl; }};class B : public A{public: B() : A() { demo_ = new int(0); } ~B() { std::cout << "~B()" << std::endl; delete demo_; }private: int *demo_;};int main(){ A * pA = new B(); delete pA; return 0;}

運行結果: 這里寫圖片描述 關于繼承的構造和析構順序的總結: 一般的繼承體系的類 基類構造 基類成員構造 子類構造 子類析構 基類成員析構 基類析構

繼承中的虛繼承

著名的菱形繼承

#define _CRT_SECURE_NO_WARNINGS#include <iostream>#include <cstring>class A{public: int a_;};class B : public A{public: int b_;};class C : public A{public: int c_;};class D : public B, public C{ // a_ b_ c_,并且a_是兩個 // 我們稱這種繼承方式為菱形繼承};int main(){ D d; d.a_; return 0;}

此時在編譯的時候就會出錯,如下所示: 這里寫圖片描述 但是,我們也是可以訪問的

#define _CRT_SECURE_NO_WARNINGS#include <iostream>#include <cstring>class A{public: int a_;};class B : public A{public: int b_;};class C : public A{public: int c_;};class D : public B, public C{ // a_ b_ c_,并且a_是兩個 // 我們稱這種繼承方式為菱形繼承};int main(){ D d; d.B::a_; d.C::a_; return 0;}

指明要訪問哪個基類中的變量,就不會出現編譯錯誤了。 這樣雖然解決了編譯錯誤的問題,但是我們不需要這樣的情況,我們只需要一份變量就可以了,那么這樣該怎么解決呢?我們可以通過虛繼承來解決這個問題?。?!

#define _CRT_SECURE_NO_WARNINGS#include <iostream>#include <cstring>class A{public: int a_;};class B : virtual public A{public: int b_;};class C : virtual public A{public: int c_;};class D : virtual public B, virtual public C{ // a_ b_ c_,并且a_是兩個 // 我們稱這種繼承方式為菱形繼承};int main(){ D d; d.a_; return 0;}

此時編譯也就可以通過了。

繼承中的純虛函數

#define _CRT_SECURE_NO_WARNINGS#include <iostream>#include <cstring>class Animal{public: virtual void Cry() = 0; // 純虛函數,無需實現,需要子類來實現};class Dog : public Animal{public:};int main(){ Dog dog; // 此時編譯不能通過,因為沒有實現基類中的純虛函數 return 0;}#define _CRT_SECURE_NO_WARNINGS#include <iostream>#include <cstring>class Animal{public: virtual void Cry() = 0; // 純虛函數,無需實現,需要子類來實現};class Dog : public Animal{public: void Cry() { }};int main(){ Dog dog; // 當實現了基類中的純虛函數后,就能實例化對象了 return 0;}#define _CRT_SECURE_NO_WARNINGS#include <iostream>#include <cstring>// 純虛函數 不能實例化class Animal{public: virtual void Cry() = 0; // 純虛函數,無需實現,需要子類來實現(強制性的進行實現)};class Dog : public Animal{public: void Cry() { }};class A : public Dog{};int main(){ Dog dog; // 當實現了基類中的純虛函數后,就能實例化對象了 // 只要有純虛函數的類,我們稱之為抽象類,抽象類是無法實例化的 A a; // 此時A類就可以實例化,因為Dog類中已經實現了純虛函數 return 0;}#define _CRT_SECURE_NO_WARNINGS#include <iostream>#include <cstring>// 純虛函數 不能實例化 是其它類的基類 經常用來作為接口使用 接口是用來進行解耦的// 接口// 沒有任何的數據成員,只是把函數名列出來class Animal{public: virtual void Cry() = 0; // 純虛函數,無需實現,需要子類來實現(強制性的進行實現) virtual void Jump() = 0; // 跳的接口};class Dog : public Animal{public: void Cry() { }};class A : public Dog{};int main(){ Dog *dog = new A(); // 用來指向子類 dog->Cry(); return 0;}
發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
高清日韩电视剧大全免费播放在线观看| 亚洲精品国精品久久99热| 国产精品h在线观看| 在线观看亚洲区| 最近中文字幕日韩精品| 欧美久久精品午夜青青大伊人| 亚洲第一页在线| 狠狠色狠色综合曰曰| 亚洲午夜久久久久久久| 亚洲少妇中文在线| 成人国产精品日本在线| 欧美大码xxxx| 久久久久久国产| 日韩高清电影免费观看完整版| 91精品在线一区| 国产剧情久久久久久| 精品中文字幕在线| 国产精品91久久久久久| 欧美激情久久久久久| 久久精品视频导航| 久久亚洲一区二区三区四区五区高| 久久久免费精品视频| 日韩精品中文字| 亚洲欧美日韩直播| 亚洲国产成人爱av在线播放| 欧美在线视频免费播放| 影音先锋日韩有码| 中文字幕九色91在线| 亚洲精品国产免费| 成人激情免费在线| 国产精品久久久久久久久免费看| 国产情人节一区| 91视频国产精品| 欧美精品久久久久久久久| 日韩h在线观看| 91成人福利在线| 国产精品久久久久7777婷婷| 秋霞成人午夜鲁丝一区二区三区| 中文字幕免费精品一区高清| 韩曰欧美视频免费观看| 日韩成人在线网站| 亚洲影院色在线观看免费| 欧美贵妇videos办公室| 国产一区二区三区中文| 日韩av电影在线播放| 亚洲欧美制服综合另类| 九色91av视频| 91精品国产色综合久久不卡98| www.99久久热国产日韩欧美.com| 国内精品久久久久影院 日本资源| 久久久久久久一区二区三区| 日韩精品在线免费观看视频| 欧美情侣性视频| 亚洲女在线观看| 国产综合在线观看视频| 国产精品成人av在线| 操91在线视频| 亚洲欧美中文日韩在线| 亚洲最大成人在线| 韩国三级日本三级少妇99| 国产精品嫩草影院一区二区| 久久99热精品这里久久精品| 91精品久久久久久久久久久久久| 日韩在线观看免费高清| 亚洲jizzjizz日本少妇| 亚洲va码欧洲m码| 欧洲一区二区视频| 日韩欧美一区二区在线| 久久国产精品久久久久久久久久| 亚洲伦理中文字幕| 91精品久久久久久久久| 欧美极品少妇xxxxⅹ免费视频| 国产福利视频一区二区| 久热精品视频在线免费观看| www.日韩视频| 97激碰免费视频| 亚洲福利视频专区| 九九热精品视频国产| 性金发美女69hd大尺寸| 日韩av网站在线| 精品久久久999| 不卡av日日日| 97国产在线视频| 精品成人av一区| 色综久久综合桃花网| 亚洲天堂一区二区三区| 国内精品久久久久影院 日本资源| 国产精品久久久久一区二区| 亚洲精品v欧美精品v日韩精品| 91精品视频一区| 久久影院资源站| 91av视频在线免费观看| 亚洲精品视频免费| 久久综合国产精品台湾中文娱乐网| 国内成人精品一区| 成人做爰www免费看视频网站| 日韩中文有码在线视频| 亚洲欧洲第一视频| 亚洲国产天堂久久综合| 久热精品视频在线观看| 亚洲欧洲在线观看| 欧美精品亚州精品| 国产精品丝袜久久久久久高清| 2021久久精品国产99国产精品| 最近的2019中文字幕免费一页| 日韩免费av片在线观看| 日韩精品有码在线观看| 国产精品自产拍在线观看| 亚洲国产中文字幕在线观看| 欧美人交a欧美精品| 国产美女扒开尿口久久久| 欧美丰满老妇厨房牲生活| 国产精品18久久久久久麻辣| 欧美美最猛性xxxxxx| 日本三级韩国三级久久| 国产精品99蜜臀久久不卡二区| 日韩美女视频中文字幕| 欧美三级欧美成人高清www| 欧美午夜视频一区二区| 久久精品色欧美aⅴ一区二区| 日本欧美国产在线| 456国产精品| 色av吧综合网| 欧美日韩激情视频| 九九精品在线播放| 久久久久久久影院| 亚洲第一区第二区| 亚洲国产精品成人va在线观看| 久久精品视频va| 国产亚洲欧洲黄色| 国产欧美日韩精品丝袜高跟鞋| 欧美视频免费在线| 国产中文欧美精品| 久久久视频精品| 精品色蜜蜜精品视频在线观看| 91精品视频免费观看| 色偷偷888欧美精品久久久| 2019av中文字幕| 亚洲天堂第一页| 国产精品7m视频| 清纯唯美亚洲激情| 亚洲精品91美女久久久久久久| 国产精品高潮呻吟久久av黑人| 国产日韩在线亚洲字幕中文| 欧美裸体xxxx极品少妇软件| 91久久精品国产91久久| 91免费国产网站| 日韩精品中文字幕久久臀| 日本久久久久久久久久久| 成人在线免费观看视视频| 久久久999成人| 日韩精品在线电影| 一区二区三区 在线观看视| 一本大道香蕉久在线播放29| 久久国产加勒比精品无码| 日本免费久久高清视频| 国产91在线播放九色快色| 国产精品入口免费视| 97精品一区二区视频在线观看| 日韩精品免费电影| 亚洲高清久久网| 国产精品偷伦一区二区| 欧美放荡办公室videos4k| 国产日韩av在线|