什么是多態(tài)?
父類指針即根據(jù)指向的不同對(duì)象,響應(yīng)同一消息(函數(shù)調(diào)用),產(chǎn)生不同行為。
多態(tài)三要素?
1,繼承
2,虛函數(shù)重寫
3,父類指針(引用)指向子類對(duì)象
多態(tài)的實(shí)現(xiàn)很簡(jiǎn)答,讓我們來看一段代碼
#include
using namespace std;
class Parent
{
public:
virtual void show()
{
cout << "我是你爹" << endl;
}
};
class Child:public Parent//1,繼承
{
public:
virtual void show()//2,虛函數(shù)重寫
{
cout << "我是你崽" << endl;
}
};
int main()
{
Parent *pa = new Child;//3,父類指針指向子類對(duì)象
pa->show();
getchar();
return 0;
}
//結(jié)果輸出的是子類的show函數(shù)--"我是你崽"
實(shí)現(xiàn)很簡(jiǎn)單,但是這又是什么原理呢?
當(dāng)我們?cè)陬愔新暶髁颂摵瘮?shù)之后,編譯器會(huì)給類添加一個(gè)vptr指針,當(dāng)定義對(duì)象的時(shí)候,會(huì)把所有虛函數(shù)放入一個(gè)叫虛函數(shù)表的順序表,然后用vptr指針指向虛函數(shù)表。當(dāng)進(jìn)行pa->show();調(diào)用的時(shí)候,C++編譯器不需要區(qū)分子類或者父類對(duì)象,只需要在pa指針中,找到vptr指針即可。
如果對(duì)象類型是子類,就調(diào)用子類的函數(shù);如果對(duì)象類型是父類,就調(diào)用父類的函數(shù),(即指向父類調(diào)父類,指向子類調(diào)子類)此為多態(tài)的表現(xiàn)。
既然類里面有vptr指針,那么我們能找到它嗎?
咱們一起來探究下:首先看下加了虛函數(shù)的類的大小有沒有變化。
可以看到加了虛函數(shù),類的大小比沒有增加虛函數(shù)的類,多了四個(gè)字節(jié)的空間,有的同學(xué)可能會(huì)說,四個(gè)字節(jié)的類型不一定是指針。不要著急,讓我們繼續(xù)往下看。
接下來我們定義對(duì)象,然后通過調(diào)試,看下局部變量窗口
從這里就可以明確看到,子類對(duì)象中有一個(gè)vptr指針,而且它是對(duì)象的第一個(gè)成員,它的類型是void**,指向的是一個(gè)順序表,下標(biāo)為0的元素裝的是我們聲明的虛函數(shù)。
那么,知道了這些,咱們能利用對(duì)象找到虛函數(shù)表,然后自己手動(dòng)調(diào)用虛函數(shù)嗎?
你們:肯定可以啊,廢話
我:。。。那就廢話不多說,歐力給!搞起
我:首先畫一張內(nèi)存模型圖,瞅瞅(畫工太丑,見諒)
1,首先,要拿到vptr指針,怎么拿呢?因?yàn)樗趯?duì)象的第一個(gè)元素,所以我們先對(duì)對(duì)象取地址&ch,這樣就拿到了對(duì)象的地址。對(duì)象的元素的內(nèi)存是連續(xù)的,但是現(xiàn)在指針的步長(zhǎng)是Child類的大小,我們需要把它當(dāng)成一個(gè)整型數(shù)組(因?yàn)関ptr是四個(gè)字節(jié)),所以需要強(qiáng)轉(zhuǎn)成int*,即(int*)&ch,這樣之后數(shù)組第一個(gè)元素就是vptr指針了,取值即可得到
(int )&ch
2,然后,前面通過調(diào)試我們知道了,vptr指針是void**類型的,所以我們也要講它轉(zhuǎn)為int*,然后取值. (int )( (int )&ch),這樣就拿到了虛函數(shù)表的第一個(gè)元素。
3,但是,現(xiàn)在拿到的元素是int*型,不是函數(shù)指針,無法調(diào)用,所以我們需要強(qiáng)轉(zhuǎn)為函數(shù)指針,才能進(jìn)行調(diào)用。
你學(xué)廢了沒?嘿嘿
-
指針
+關(guān)注
關(guān)注
1文章
484瀏覽量
70998 -
對(duì)象
+關(guān)注
關(guān)注
1文章
38瀏覽量
17506 -
函數(shù)調(diào)用
+關(guān)注
關(guān)注
0文章
19瀏覽量
2650
發(fā)布評(píng)論請(qǐng)先 登錄
C++的多態(tài)詳解
多態(tài)子VI已斷開,運(yùn)行該VI前必須解決多態(tài)子VI的所有錯(cuò)誤。
LabVIEW多態(tài) VI (Polymorphic VIs)的使用
基于Java多態(tài)性的應(yīng)用程序設(shè)計(jì)
什么是方法的重載(多態(tài)性)?

C++的動(dòng)態(tài)多態(tài)和靜態(tài)多態(tài)

java多態(tài)性的實(shí)現(xiàn)
多態(tài)路由機(jī)制研究

JAVA教程之消息、繼承與多態(tài)的應(yīng)用和資料介紹說明

Java中的多態(tài)有哪些表現(xiàn)形式
java的封裝繼承和多態(tài)

評(píng)論