C++基類指針和派生類指針之間的轉(zhuǎn)換方法講解
函數(shù)重載、函數(shù)隱藏、函數(shù)覆蓋
函數(shù)重載只會發(fā)生在同作用域中(或同一個類中),函數(shù)名稱相同,但參數(shù)類型或參數(shù)個數(shù)不同。 函數(shù)重載不能通過函數(shù)的返回類型來區(qū)分,因為在函數(shù)返回之前我們并不知道函數(shù)的返回類型。
函數(shù)隱藏和函數(shù)覆蓋只會發(fā)生在基類和派生類之間。
函數(shù)隱藏是指派生類中函數(shù)與基類中的函數(shù)同名,但是這個函數(shù)在基類中并沒有被定義為虛函數(shù),這種情況就是函數(shù)的隱藏。
所謂隱藏是指使用常規(guī)的調(diào)用方法,派生類對象訪問這個函數(shù)時,會優(yōu)先訪問派生類中的這個函數(shù),基類中的這個函數(shù)對派生類對象來說是隱藏起來的。 但是隱藏并不意味這不存在或完全不可訪問。通過 b->Base::func()訪問基類中被隱藏的函數(shù)。
函數(shù)覆蓋特指由基類中定義的虛函數(shù)引發(fā)的一種多態(tài)現(xiàn)象。在某基類中聲明為 virtual 并在一個或多個派生類中被重新定義的成員函數(shù),用法格式為:virtual 函數(shù)返回類型 函數(shù)名(參數(shù)表) {函數(shù)體};實現(xiàn)多態(tài)性,通過指向派生類的基類指針或引用,訪問派生類中同名覆蓋成員函數(shù)。
函數(shù)覆蓋的條件:
- 1:基類中的成員函數(shù)被virtual關(guān)鍵字聲明為虛函數(shù);
- 2:派生類中該函數(shù)必須和基類中函數(shù)的名稱、參數(shù)類型和個數(shù)等完全一致;
- 3:將派生類的對象賦給基類指針或者引用,實現(xiàn)多態(tài)。
函數(shù)覆蓋(多態(tài))實現(xiàn)了一種基類訪問(不同)派生類的方法。我們把它稱為基類的逆襲。
基類指針和派生類指針之間的轉(zhuǎn)換
1. 基類指針指向基類對象、派生類指針指向派生類對象
這種情況是常用的,只需要通過對應(yīng)類的指針直接調(diào)用對應(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. 基類指針指向派生類對象
這種情況是允許的,通過定義一個基類指針和一個派生類對象,把基類指針指向派生類對象,但是需要注意,通常情況這時的指針調(diào)用的是基類的成員函數(shù)。分四種情況:
一、 函數(shù)在基類和派生類中都存在
這時通過“指向派生類對象的基類指針”調(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ù),則編譯器會報錯。
error C2039: “xxx”: 不是“Father”的成員
三、 將基類指針強制轉(zhuǎn)換為派生類指針
這種是向下的強制類型轉(zhuǎn)換,轉(zhuǎn)換之后“指向派生類的基類指針”就可以訪問派生類的成員函數(shù):
Son s1;
Father* f = &s1;
Son *s = (Son*)f;
s->print1(); //調(diào)用派生類成員函數(shù)
但是這種強制轉(zhuǎn)換操作是一種潛在的危險操作。
四、基類中存在虛函數(shù)的情況
如果基類中的成員函數(shù)被定義為虛函數(shù),并且在派生類中也實現(xiàn)了該函數(shù),則通過“指向派生類的基類指針” 訪問虛函數(shù),訪問的是派生類中的實現(xiàn)。允許“基類指針指向派生類”這個操作,最大的意義也就在此,通過虛函數(shù)和函數(shù)覆蓋,實現(xiàn)了“多態(tài)”(指向不同的派生類,實現(xiàn)不同功能)。
Father f1;
Son s1;
Father* f = &s1;
f->print(); //調(diào)用派生類成員函數(shù)
3. 派生類指針指向基類對象
會產(chǎn)生編譯錯誤?;悓ο鬅o法被當(dāng)作派生類對象,派生類中可能具有只有派生類才有的成員或成員函數(shù)。
即便是使用強制轉(zhuǎn)換,將派生類指針強制轉(zhuǎn)換成基類指針,通過這個“強制指向基類的派生類指針”訪問的函數(shù)依然是派生類的成員函數(shù)。
Father f1;
Son s1;
Son* s=&s1;
Father* f = (Father*) s;
f->print(); //調(diào)用派生類成員函數(shù)
綜上,可以通過基類指針訪問派生類方法(強制轉(zhuǎn)換和虛函數(shù)),不存在通過派生類指針調(diào)用基類成員函數(shù)的方法(即便是強制轉(zhuǎn)換)。
總結(jié)
以上就是這篇文章的全部內(nèi)容了,希望本文的內(nèi)容對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,謝謝大家對我們的支持。如果你想了解更多相關(guān)內(nèi)容請查看下面相關(guān)鏈接
上一篇:詳解C++ 臨時量與臨時對象及程序的相關(guān)優(yōu)化
欄 目:C語言
本文標(biāo)題:C++基類指針和派生類指針之間的轉(zhuǎn)換方法講解
本文地址:http://mengdiqiu.com.cn/a1/Cyuyan/342.html
您可能感興趣的文章
- 04-02c語言沒有round函數(shù) round c語言
- 01-10深入理解C++中常見的關(guān)鍵字含義
- 01-10使用C++實現(xiàn)全排列算法的方法詳解
- 01-10c++中inline的用法分析
- 01-10用C++實現(xiàn)DBSCAN聚類算法
- 01-10全排列算法的非遞歸實現(xiàn)與遞歸實現(xiàn)的方法(C++)
- 01-10C++大數(shù)模板(推薦)
- 01-10淺談C/C++中的static與extern關(guān)鍵字的使用詳解
- 01-10深入C/C++浮點數(shù)在內(nèi)存中的存儲方式詳解
- 01-10深入理解數(shù)組指針與指針數(shù)組的區(qū)別


閱讀排行
本欄相關(guān)
- 04-02c語言函數(shù)調(diào)用后清空內(nèi)存 c語言調(diào)用
- 04-02func函數(shù)+在C語言 func函數(shù)在c語言中
- 04-02c語言的正則匹配函數(shù) c語言正則表達
- 04-02c語言用函數(shù)寫分段 用c語言表示分段
- 04-02c語言中對數(shù)函數(shù)的表達式 c語言中對
- 04-02c語言編寫函數(shù)冒泡排序 c語言冒泡排
- 04-02c語言沒有round函數(shù) round c語言
- 04-02c語言分段函數(shù)怎么求 用c語言求分段
- 04-02C語言中怎么打出三角函數(shù) c語言中怎
- 04-02c語言調(diào)用函數(shù)求fibo C語言調(diào)用函數(shù)求
隨機閱讀
- 04-02jquery與jsp,用jquery
- 01-10SublimeText編譯C開發(fā)環(huán)境設(shè)置
- 08-05DEDE織夢data目錄下的sessions文件夾有什
- 01-10C#中split用法實例總結(jié)
- 01-11Mac OSX 打開原生自帶讀寫NTFS功能(圖文
- 01-10delphi制作wav文件的方法
- 08-05織夢dedecms什么時候用欄目交叉功能?
- 01-10使用C語言求解撲克牌的順子及n個骰子
- 01-11ajax實現(xiàn)頁面的局部加載
- 08-05dedecms(織夢)副欄目數(shù)量限制代碼修改