深入了解C++的多態(tài)與虛函數(shù)
1.多態(tài)的機(jī)制與虛函數(shù)的機(jī)制
1.1 多態(tài)的機(jī)制
1.當(dāng)在類中使用virtual聲明一個(gè)函數(shù)為虛函數(shù)時(shí),在編譯時(shí),編譯器會(huì)自動(dòng)在基類中默默地安插一個(gè)虛函數(shù)表指針,同時(shí)的.rodata段為這類生成一張?zhí)摵瘮?shù)表,用來(lái)保存類中的虛函數(shù)的地址。
2.當(dāng)繼承發(fā)生時(shí),父類中的虛指針就被子類給繼承了下來(lái),所以他的類對(duì)象空間就增大了一個(gè)指針的大小。
3.當(dāng)子類構(gòu)造對(duì)象時(shí),這根繼承而來(lái)的虛指針,將會(huì)在子類的構(gòu)造函數(shù)中被重新賦值,所賦的值即為子類類中產(chǎn)生的虛函數(shù)表地址。
4.當(dāng)使用父類指針或引用,對(duì)虛函數(shù)進(jìn)行調(diào)用時(shí),通過(guò)這個(gè)虛函數(shù)表指針,在虛函數(shù)表中查找虛函數(shù)的地址,從而調(diào)用不同類的虛函數(shù)。
1.2 虛函數(shù)的機(jī)制
虛函數(shù)的意義何在?就是用來(lái)承接動(dòng)態(tài)多態(tài)的。他是如何承接這種動(dòng)態(tài)多態(tài)機(jī)制的呢?
當(dāng)子類之中函數(shù)與父類之中的虛函數(shù)重名時(shí),且返回值與形參列表都一致時(shí),將是對(duì)父類虛函數(shù)的重寫。當(dāng)在子類重寫虛函數(shù)時(shí),將會(huì)把虛函數(shù)表中的父類的虛函數(shù)地址覆蓋掉。
1.3虛函數(shù)表的結(jié)構(gòu)圖
1.4 動(dòng)態(tài)多態(tài)實(shí)現(xiàn)的三個(gè)前提件(很重要)
1.有繼承關(guān)系
2.基類中有虛函數(shù),且子類重寫虛函數(shù)
3.基類指針或引用,指向或引用父類對(duì)象,就會(huì)形成動(dòng)態(tài)多態(tài)
2.多態(tài)實(shí)例應(yīng)用
#include <iostream> using namespace std; class Driver{ public: virtual void show_info() { cout<<"我是司機(jī)"<<endl; } }; class Bwm:public Driver { public: void show_info() { cout<<"我開(kāi)的是寶馬"<<endl; } }; class Benchi:public Driver { public: void show_info() { cout<<"我開(kāi)的是奔馳"<<endl; } }; class Tuolaji:public Driver { public: void show_info() { cout<<"我開(kāi)的是拖拉機(jī)"<<endl; } }; class Kai { public: void kaiche(Driver& p) { p.show_info(); } }; int main() { Bwm bwm; Tuolaji tuolaji; Benchi benchi; Kai p; p.kaiche(tuolaji); p.kaiche(benchi); p.kaiche(benchi); return 0; }
結(jié)果圖:
3.多態(tài)的巨大問(wèn)題與虛析構(gòu)
3.1代碼舉例說(shuō)明
#include <iostream> using namespace std; class A { public: A() { cout<<"A的構(gòu)造"<<endl; } ~A() { cout<<"A的析構(gòu)"<<endl; } virtual void show_info() { cout<<"愛(ài)吃飯"<<endl; } }; class B:public A { public: B() { cout<<"B的構(gòu)造"<<endl; } ~B() { cout<<"B的析構(gòu)"<<endl; } void show_info() { cout<<"愛(ài)吃糖"<<endl; } }; int main() { A* a=new B; a->show_info(); delete a; return 0; }
結(jié)果圖:
由圖可知:當(dāng)用虛函數(shù)實(shí)現(xiàn)多態(tài)的時(shí)候,子類的的析構(gòu)無(wú)法進(jìn)行。
所以我們應(yīng)該怎么解決呢?
3.2代碼實(shí)現(xiàn)
#include <iostream> using namespace std; class A { public: A() { cout<<"A的構(gòu)造"<<endl; } virtual~A() { cout<<"A的析構(gòu)"<<endl; } virtual void show_info() { cout<<"愛(ài)吃飯"<<endl; } }; class B:public A { public: B() { cout<<"B的構(gòu)造"<<endl; } ~B() { cout<<"B的析構(gòu)"<<endl; } void show_info() { cout<<"愛(ài)吃糖"<<endl; } }; int main() { A* a=new B; a->show_info(); delete a; return 0; }
結(jié)果圖:
所以,為了避免子類的析構(gòu)無(wú)法執(zhí)行而造成的內(nèi)存泄漏問(wèn)題,應(yīng)該把最遠(yuǎn)端父類的析構(gòu)函數(shù)定義為虛析構(gòu)。虛析構(gòu)的語(yǔ)法即是在最遠(yuǎn)端父類的析構(gòu)函數(shù)名前 加virtual進(jìn)行修飾即可。
虛析構(gòu)如下:
virtual~A() { //虛析構(gòu)的定義形式。 }
4.純虛函數(shù)與抽象類
4.1純虛函數(shù)語(yǔ)法格式
class + 類名
{
public:
virtual void showInfo() = 0;
};
4.2純虛函數(shù)的定義
在類中定義一個(gè)沒(méi)有函數(shù)體的虛函數(shù)就叫做純虛函數(shù)。類中有純虛函數(shù)的類就叫做抽象類。抽象類被人稱接口類。這個(gè)純虛函數(shù)只有一個(gè)函數(shù)名做為函數(shù)功能的表現(xiàn),而沒(méi)有函數(shù)體的實(shí)現(xiàn)。純虛函數(shù)必須在子類之中進(jìn)行實(shí)現(xiàn),如果繼承的子類沒(méi)有實(shí)現(xiàn)父類的純虛函數(shù),那么這個(gè)子類也將成員抽象類。抽象類是不可以定義對(duì)象的。純虛函數(shù)也叫接口類中的接口。
4.3抽象類的應(yīng)用實(shí)例
#include <iostream> using namespace std; class A { public: virtual void show_info()=0; virtual void goshopping()=0; }; class B:public A { public: void show_info() { cout<<"我是大哥"<<endl; } void goshopping() { cout<<"我是小弟"<<endl; } }; int main() { A* a=new B; a->show_info(); a->goshopping(); return 0; }
到此這篇關(guān)于深入了解C++的多態(tài)與虛函數(shù)的文章就介紹到這了,更多相關(guān)C++多態(tài) 虛函數(shù)內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
判斷一個(gè)數(shù)是不是素?cái)?shù)的方法
判斷一個(gè)數(shù)是不是素?cái)?shù)的方法,需要的朋友可以參考一下2013-03-03C++ 回調(diào)接口設(shè)計(jì)和二進(jìn)制兼容詳細(xì)
再開(kāi)發(fā)視頻編輯 SDK,SDK的回調(diào)接口設(shè)計(jì)成 C 風(fēng)格,結(jié)構(gòu)中放著一些函數(shù)指針,既然對(duì)外接口是 C++,為什么不直接使用 C++ 的虛函數(shù)?這篇文章便對(duì)這一問(wèn)題做個(gè)詳細(xì)介紹,需要的朋友可以參考一下2021-09-09C語(yǔ)言指針變量作為函數(shù)參數(shù)的實(shí)現(xiàn)步驟詳解
這篇文章主要介紹了C語(yǔ)言指針變量作為函數(shù)參數(shù)的實(shí)現(xiàn)步驟,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)吧2023-02-02Visual?Studio?2022?激活碼(親測(cè)可用)
在?Visual?Studio?2019?的基礎(chǔ)上,新版集成開(kāi)發(fā)壞境提供了非常多的改進(jìn),包括對(duì)?64?位、.NET?6?的支持,為核心調(diào)試器提供更好的性能。本文給大家分享Visual?Studio?2022?激活碼,需要的朋友參考下吧2021-12-12C++實(shí)踐分?jǐn)?shù)類中運(yùn)算符重載的方法參考
今天小編就為大家分享一篇關(guān)于C++實(shí)踐分?jǐn)?shù)類中運(yùn)算符重載的方法參考,小編覺(jué)得內(nèi)容挺不錯(cuò)的,現(xiàn)在分享給大家,具有很好的參考價(jià)值,需要的朋友一起跟隨小編來(lái)看看吧2019-02-02C++實(shí)現(xiàn)LeetCode(769.可排序的最大塊數(shù))
這篇文章主要介紹了C++實(shí)現(xiàn)LeetCode(769.可排序的最大塊數(shù)),本篇文章通過(guò)簡(jiǎn)要的案例,講解了該項(xiàng)技術(shù)的了解與使用,以下就是詳細(xì)內(nèi)容,需要的朋友可以參考下2021-07-07