C++基類(lèi)指針和派生類(lèi)指針之間的轉(zhuǎn)換方法講解
函數(shù)重載、函數(shù)隱藏、函數(shù)覆蓋
函數(shù)重載只會(huì)發(fā)生在同作用域中(或同一個(gè)類(lèi)中),函數(shù)名稱(chēng)相同,但參數(shù)類(lèi)型或參數(shù)個(gè)數(shù)不同。 函數(shù)重載不能通過(guò)函數(shù)的返回類(lèi)型來(lái)區(qū)分,因?yàn)樵诤瘮?shù)返回之前我們并不知道函數(shù)的返回類(lèi)型。
函數(shù)隱藏和函數(shù)覆蓋只會(huì)發(fā)生在基類(lèi)和派生類(lèi)之間。
函數(shù)隱藏是指派生類(lèi)中函數(shù)與基類(lèi)中的函數(shù)同名,但是這個(gè)函數(shù)在基類(lèi)中并沒(méi)有被定義為虛函數(shù),這種情況就是函數(shù)的隱藏。
所謂隱藏是指使用常規(guī)的調(diào)用方法,派生類(lèi)對(duì)象訪(fǎng)問(wèn)這個(gè)函數(shù)時(shí),會(huì)優(yōu)先訪(fǎng)問(wèn)派生類(lèi)中的這個(gè)函數(shù),基類(lèi)中的這個(gè)函數(shù)對(duì)派生類(lèi)對(duì)象來(lái)說(shuō)是隱藏起來(lái)的。 但是隱藏并不意味這不存在或完全不可訪(fǎng)問(wèn)。通過(guò) b->Base::func()訪(fǎng)問(wèn)基類(lèi)中被隱藏的函數(shù)。
函數(shù)覆蓋特指由基類(lèi)中定義的虛函數(shù)引發(fā)的一種多態(tài)現(xiàn)象。在某基類(lèi)中聲明為 virtual 并在一個(gè)或多個(gè)派生類(lèi)中被重新定義的成員函數(shù),用法格式為:virtual 函數(shù)返回類(lèi)型 函數(shù)名(參數(shù)表) {函數(shù)體};實(shí)現(xiàn)多態(tài)性,通過(guò)指向派生類(lèi)的基類(lèi)指針或引用,訪(fǎng)問(wèn)派生類(lèi)中同名覆蓋成員函數(shù)。
函數(shù)覆蓋的條件:
- 1:基類(lèi)中的成員函數(shù)被virtual關(guān)鍵字聲明為虛函數(shù);
- 2:派生類(lèi)中該函數(shù)必須和基類(lèi)中函數(shù)的名稱(chēng)、參數(shù)類(lèi)型和個(gè)數(shù)等完全一致;
- 3:將派生類(lèi)的對(duì)象賦給基類(lèi)指針或者引用,實(shí)現(xiàn)多態(tài)。
函數(shù)覆蓋(多態(tài))實(shí)現(xiàn)了一種基類(lèi)訪(fǎng)問(wèn)(不同)派生類(lèi)的方法。我們把它稱(chēng)為基類(lèi)的逆襲。
基類(lèi)指針和派生類(lèi)指針之間的轉(zhuǎn)換
1. 基類(lèi)指針指向基類(lèi)對(duì)象、派生類(lèi)指針指向派生類(lèi)對(duì)象
這種情況是常用的,只需要通過(guò)對(duì)應(yīng)類(lèi)的指針直接調(diào)用對(duì)應(yīng)類(lèi)的功能就可以了。
#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. 基類(lèi)指針指向派生類(lèi)對(duì)象
這種情況是允許的,通過(guò)定義一個(gè)基類(lèi)指針和一個(gè)派生類(lèi)對(duì)象,把基類(lèi)指針指向派生類(lèi)對(duì)象,但是需要注意,通常情況這時(shí)的指針調(diào)用的是基類(lèi)的成員函數(shù)。分四種情況:
一、 函數(shù)在基類(lèi)和派生類(lèi)中都存在
這時(shí)通過(guò)“指向派生類(lèi)對(duì)象的基類(lèi)指針”調(diào)用成員函數(shù),調(diào)用的是基類(lèi)的成員函數(shù)。
Father f1;
Son s1;
Father* f = &s1;
f->print(); //調(diào)用的是基類(lèi)成員函數(shù)
二、函數(shù)在基類(lèi)中不存在,在派生類(lèi)中存在
由于調(diào)用的還是基類(lèi)中的成員函數(shù),試圖通過(guò)基類(lèi)指針調(diào)用派生類(lèi)才有的成員函數(shù),則編譯器會(huì)報(bào)錯(cuò)。
error C2039: “xxx”: 不是“Father”的成員
三、 將基類(lèi)指針強(qiáng)制轉(zhuǎn)換為派生類(lèi)指針
這種是向下的強(qiáng)制類(lèi)型轉(zhuǎn)換,轉(zhuǎn)換之后“指向派生類(lèi)的基類(lèi)指針”就可以訪(fǎng)問(wèn)派生類(lèi)的成員函數(shù):
Son s1;
Father* f = &s1;
Son *s = (Son*)f;
s->print1(); //調(diào)用派生類(lèi)成員函數(shù)
但是這種強(qiáng)制轉(zhuǎn)換操作是一種潛在的危險(xiǎn)操作。
四、基類(lèi)中存在虛函數(shù)的情況
如果基類(lèi)中的成員函數(shù)被定義為虛函數(shù),并且在派生類(lèi)中也實(shí)現(xiàn)了該函數(shù),則通過(guò)“指向派生類(lèi)的基類(lèi)指針” 訪(fǎng)問(wèn)虛函數(shù),訪(fǎng)問(wèn)的是派生類(lèi)中的實(shí)現(xiàn)。允許“基類(lèi)指針指向派生類(lèi)”這個(gè)操作,最大的意義也就在此,通過(guò)虛函數(shù)和函數(shù)覆蓋,實(shí)現(xiàn)了“多態(tài)”(指向不同的派生類(lèi),實(shí)現(xiàn)不同功能)。
Father f1;
Son s1;
Father* f = &s1;
f->print(); //調(diào)用派生類(lèi)成員函數(shù)
3. 派生類(lèi)指針指向基類(lèi)對(duì)象
會(huì)產(chǎn)生編譯錯(cuò)誤。基類(lèi)對(duì)象無(wú)法被當(dāng)作派生類(lèi)對(duì)象,派生類(lèi)中可能具有只有派生類(lèi)才有的成員或成員函數(shù)。
即便是使用強(qiáng)制轉(zhuǎn)換,將派生類(lèi)指針強(qiáng)制轉(zhuǎn)換成基類(lèi)指針,通過(guò)這個(gè)“強(qiáng)制指向基類(lèi)的派生類(lèi)指針”訪(fǎng)問(wèn)的函數(shù)依然是派生類(lèi)的成員函數(shù)。
Father f1;
Son s1;
Son* s=&s1;
Father* f = (Father*) s;
f->print(); //調(diào)用派生類(lèi)成員函數(shù)
綜上,可以通過(guò)基類(lèi)指針訪(fǎng)問(wèn)派生類(lèi)方法(強(qiáng)制轉(zhuǎn)換和虛函數(shù)),不存在通過(guò)派生類(lèi)指針調(diào)用基類(lèi)成員函數(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)文章
C++封裝靜態(tài)鏈接庫(kù)和使用的詳細(xì)步驟
這篇文章主要介紹了C++封裝靜態(tài)鏈接庫(kù)和使用,本文描述了怎么去把一個(gè)C++程序封裝成一個(gè)靜態(tài)庫(kù)并且如何去使用這些靜態(tài)庫(kù),需要的朋友可以參考下2022-08-08
如何判斷一個(gè)整數(shù)的二進(jìn)制中有多少個(gè)1
本篇文章是對(duì)如何判斷一個(gè)整數(shù)的二進(jìn)制中有多少個(gè)1的方法進(jìn)行了詳細(xì)的分析介紹,需要的朋友參考下2013-05-05
C++的靜態(tài)成員變量和靜態(tài)成員函數(shù)你了解多少
這篇文章主要為大家詳細(xì)介紹了C++的靜態(tài)成員變量和靜態(tài)成員函數(shù),文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-02-02
C語(yǔ)言實(shí)現(xiàn)學(xué)生信息管理系統(tǒng)(鏈表)
這篇文章主要為大家詳細(xì)介紹了C語(yǔ)言實(shí)現(xiàn)學(xué)生信息管理系統(tǒng),文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-06-06
C++中的結(jié)構(gòu)體vector排序問(wèn)題
這篇文章主要介紹了C++中的結(jié)構(gòu)體vector排序問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-11-11
FFmpeg實(shí)現(xiàn)多線(xiàn)程編碼并保存mp4文件
這篇文章主要為大家介紹了FFmpeg如何持續(xù)的從指定內(nèi)存中讀取原始數(shù)據(jù),再將解碼數(shù)據(jù)存入隊(duì)列中,并通過(guò)單獨(dú)的線(xiàn)程進(jìn)行編碼,最后保存為mp4文件,感興趣的可以了解下2023-08-08

