C++基類指針和派生類指針之間的轉(zhuǎn)換方法講解
函數(shù)重載、函數(shù)隱藏、函數(shù)覆蓋
函數(shù)重載只會(huì)發(fā)生在同作用域中(或同一個(gè)類中),函數(shù)名稱相同,但參數(shù)類型或參數(shù)個(gè)數(shù)不同。 函數(shù)重載不能通過函數(shù)的返回類型來區(qū)分,因?yàn)樵诤瘮?shù)返回之前我們并不知道函數(shù)的返回類型。
函數(shù)隱藏和函數(shù)覆蓋只會(huì)發(fā)生在基類和派生類之間。
函數(shù)隱藏是指派生類中函數(shù)與基類中的函數(shù)同名,但是這個(gè)函數(shù)在基類中并沒有被定義為虛函數(shù),這種情況就是函數(shù)的隱藏。
所謂隱藏是指使用常規(guī)的調(diào)用方法,派生類對(duì)象訪問這個(gè)函數(shù)時(shí),會(huì)優(yōu)先訪問派生類中的這個(gè)函數(shù),基類中的這個(gè)函數(shù)對(duì)派生類對(duì)象來說是隱藏起來的。 但是隱藏并不意味這不存在或完全不可訪問。通過 b->Base::func()訪問基類中被隱藏的函數(shù)。
函數(shù)覆蓋特指由基類中定義的虛函數(shù)引發(fā)的一種多態(tài)現(xiàn)象。在某基類中聲明為 virtual 并在一個(gè)或多個(gè)派生類中被重新定義的成員函數(shù),用法格式為:virtual 函數(shù)返回類型 函數(shù)名(參數(shù)表) {函數(shù)體};實(shí)現(xiàn)多態(tài)性,通過指向派生類的基類指針或引用,訪問派生類中同名覆蓋成員函數(shù)。
函數(shù)覆蓋的條件:
- 1:基類中的成員函數(shù)被virtual關(guān)鍵字聲明為虛函數(shù);
- 2:派生類中該函數(shù)必須和基類中函數(shù)的名稱、參數(shù)類型和個(gè)數(shù)等完全一致;
- 3:將派生類的對(duì)象賦給基類指針或者引用,實(shí)現(xiàn)多態(tài)。
函數(shù)覆蓋(多態(tài))實(shí)現(xiàn)了一種基類訪問(不同)派生類的方法。我們把它稱為基類的逆襲。
基類指針和派生類指針之間的轉(zhuǎn)換
1. 基類指針指向基類對(duì)象、派生類指針指向派生類對(duì)象
這種情況是常用的,只需要通過對(duì)應(yīng)類的指針直接調(diào)用對(duì)應(yīng)類的功能就可以了。
#include<iostream> using namespace std; class Father{ public: void print() { printf("Father's function!"); } }; class Son:public Father { public: void print() { printf("Son's function!"); } }; int main() { Father f1; Son s1; Father* f = &f1; Son* s = &s1; f->print(); cout<<endl<<endl; s->print(); }
2. 基類指針指向派生類對(duì)象
這種情況是允許的,通過定義一個(gè)基類指針和一個(gè)派生類對(duì)象,把基類指針指向派生類對(duì)象,但是需要注意,通常情況這時(shí)的指針調(diào)用的是基類的成員函數(shù)。分四種情況:
一、 函數(shù)在基類和派生類中都存在
這時(shí)通過“指向派生類對(duì)象的基類指針”調(diào)用成員函數(shù),調(diào)用的是基類的成員函數(shù)。
Father f1;
Son s1;
Father* f = &s1;
f->print(); //調(diào)用的是基類成員函數(shù)
二、函數(shù)在基類中不存在,在派生類中存在
由于調(diào)用的還是基類中的成員函數(shù),試圖通過基類指針調(diào)用派生類才有的成員函數(shù),則編譯器會(huì)報(bào)錯(cuò)。
error C2039: “xxx”: 不是“Father”的成員
三、 將基類指針強(qiáng)制轉(zhuǎn)換為派生類指針
這種是向下的強(qiáng)制類型轉(zhuǎn)換,轉(zhuǎn)換之后“指向派生類的基類指針”就可以訪問派生類的成員函數(shù):
Son s1;
Father* f = &s1;
Son *s = (Son*)f;
s->print1(); //調(diào)用派生類成員函數(shù)
但是這種強(qiáng)制轉(zhuǎn)換操作是一種潛在的危險(xiǎn)操作。
四、基類中存在虛函數(shù)的情況
如果基類中的成員函數(shù)被定義為虛函數(shù),并且在派生類中也實(shí)現(xiàn)了該函數(shù),則通過“指向派生類的基類指針” 訪問虛函數(shù),訪問的是派生類中的實(shí)現(xiàn)。允許“基類指針指向派生類”這個(gè)操作,最大的意義也就在此,通過虛函數(shù)和函數(shù)覆蓋,實(shí)現(xiàn)了“多態(tài)”(指向不同的派生類,實(shí)現(xiàn)不同功能)。
Father f1;
Son s1;
Father* f = &s1;
f->print(); //調(diào)用派生類成員函數(shù)
3. 派生類指針指向基類對(duì)象
會(huì)產(chǎn)生編譯錯(cuò)誤。基類對(duì)象無法被當(dāng)作派生類對(duì)象,派生類中可能具有只有派生類才有的成員或成員函數(shù)。
即便是使用強(qiáng)制轉(zhuǎn)換,將派生類指針強(qiáng)制轉(zhuǎn)換成基類指針,通過這個(gè)“強(qiáng)制指向基類的派生類指針”訪問的函數(shù)依然是派生類的成員函數(shù)。
Father f1;
Son s1;
Son* s=&s1;
Father* f = (Father*) s;
f->print(); //調(diào)用派生類成員函數(shù)
綜上,可以通過基類指針訪問派生類方法(強(qiáng)制轉(zhuǎn)換和虛函數(shù)),不存在通過派生類指針調(diào)用基類成員函數(shù)的方法(即便是強(qiáng)制轉(zhuǎn)換)。
總結(jié)
以上就是這篇文章的全部?jī)?nèi)容了,希望本文的內(nèi)容對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,謝謝大家對(duì)腳本之家的支持。如果你想了解更多相關(guān)內(nèi)容請(qǐng)查看下面相關(guān)鏈接
相關(guān)文章
Matlab實(shí)現(xiàn)好看的配對(duì)箱線圖的繪制
配對(duì)箱線圖,常見于配對(duì)樣本的數(shù)據(jù)分析中,它除了能夠表現(xiàn)兩組的整體差異,還能夠清晰地呈現(xiàn)單個(gè)樣本的前后改變。本文將用Matlab實(shí)現(xiàn)配對(duì)箱線圖的繪制,需要的可以參考一下2022-08-08C++?LeetCode1769移動(dòng)所有球到每個(gè)盒子最小操作數(shù)示例
這篇文章主要為大家介紹了C++?LeetCode1769移動(dòng)所有球到每個(gè)盒子所需最小操作數(shù)示例,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-12-12使用C++實(shí)現(xiàn)監(jiān)控文件是否被修改
軟件開發(fā)過程中經(jīng)常會(huì)用到配置文件,某些應(yīng)用場(chǎng)景要求在軟件運(yùn)行時(shí)動(dòng)態(tài)修改配置文件,此時(shí)就需要監(jiān)控配置文件是否被修改,下面我們就來看看如何使用C++實(shí)現(xiàn)這一功能吧2024-02-02C語(yǔ)言如何利用異或進(jìn)行兩個(gè)值的交換詳解
最近在工作中遇到了兩個(gè)值交換的需求,發(fā)現(xiàn)自己對(duì)異或有些忘記,所以索性寫出來,方便以后需要的時(shí)候參考學(xué)習(xí),下面這篇文章主要給大家介紹了關(guān)于C語(yǔ)言如何利用異或進(jìn)行兩個(gè)值的交換的相關(guān)資料,需要的朋友可以參考下。2017-09-09Qt中圖片旋轉(zhuǎn)縮放操作的實(shí)現(xiàn)
本文主要介紹了Qt中圖片旋轉(zhuǎn)縮放操作的實(shí)現(xiàn),文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2024-01-01