c++中為什么可以通過指針或引用實現(xiàn)多態(tài)詳解
引言:
在c++中司空見慣的事情就是:可以通過指針和引用可以實現(xiàn)多態(tài),而對象不可以。 那為什么?讓我們來解開這神秘的暗紗!
1、 類對象的存儲方式:
在一個類的實例中,只會存放非靜態(tài)的成員變量。 如果該類中存在虛函數(shù)的話,再多加一個指向虛函數(shù)列表指針—vptr。
例如聲明如下兩個類,并分別實例化兩個對象,它們的內(nèi)存分配大致如下:(vptr具體在什么位置,與編譯器有關(guān),大多數(shù)都在開始處)
class base { public: virtual ~base() {}; virtual string GetName() { return "base"; } GetA(); int a; }; class derived : public base { public: virtual ~derived() {}; virtual string GetName() { return "derived";} GetB(); int b; };base B1, B2;derived D1, D2;
內(nèi)存分布大致如下:
1. 類對象中,只有成員變量與vptr.
2. 普通成員函數(shù)在內(nèi)存的某一位置放著。它們與c語言中定義的普通函數(shù)沒有區(qū)別。 當我們通過對象或?qū)ο笾羔樥{(diào)用普通成員函數(shù)時, 編譯器會拿到它。怎么拿到呢?當然是通過名字了,編譯器都會對我們寫的函數(shù)的名字進行修飾映射,讓它們變成內(nèi)存中唯一的函數(shù)名。
3. 無論基類還是子類,每一種類類型的虛函數(shù)表只有一份,它里面存放了基類的類型信息和指向基類中的虛函數(shù)的指針。 某一類類型的所有對象都指向了相同的虛函數(shù)表。
2. 無論通過對象還是指針,能使用的方法只與它們靜態(tài)類型有關(guān)。
例如:下面的 base類型的對象B1或指針pB1,只能使用GetName() 和GetA()方法。 無論它們是如何來的?。。。?!
// 直接構(gòu)造得到 base B1; base* pB1 = new base(); // 即使從子類轉(zhuǎn)換而來, 通過B1或pB1也永遠訪問不到GetB()方法。 derived d1; B1 = d1; pB1 = new derived();
3. 不同類型的指針有什么區(qū)別?
本質(zhì)上它們沒有任何區(qū)別,在32/64位系統(tǒng)中都是4/8字節(jié)的一個變量。 唯一不同的就是編譯器解釋它們的方式,即通過指針來尋址出來的對象類型不同,大小不同 ,指針類型來告訴編譯器如何解釋該指針。
4. 指針與引用來實現(xiàn)多態(tài)
有代碼如下 :
derived* _pD = new derived(); base* _pB = _pD; _pB.GetName(); // 返回 derived.
想要知道如何通過指針來實現(xiàn)的多態(tài),就要看看對基類指針賦值是發(fā)生了什么! 具體來說 如下圖所示:
我們會發(fā)現(xiàn),對指針的賦值,僅僅是讓基類指針_pB指向的子類對象的地地址。 當我們使用基類指針調(diào)用GetName()函數(shù)(該函數(shù)是虛函數(shù),它的地址在函數(shù)表中)時, 會由_pB指向的地址找到子類的虛函數(shù)表指針vptr_上海,再由vptr_上海在虛函數(shù)表中找到子類的GetName(),從而調(diào)用它。就這樣實現(xiàn)了多態(tài)。
5. 對象不能實現(xiàn)多態(tài)
有代碼如下:
base B1; derived D1; B1 = D1; B1.GetName(); // 返回 base base B2 = D1 B2.GetName(); // 返回 base
上面代碼中無論賦值操作還是賦值構(gòu)造時, 只會處理成員變量,一個類對象里面的vptr永遠不會變,永遠都會指向所屬類型的虛函數(shù)表,操作如下圖所示:
因此,通過對象調(diào)用虛函數(shù)時,就沒有必要進行動態(tài)解析了,白白增加了間接性,浪費性能。編譯器直接在編譯時就可以確認具體調(diào)用哪一個函數(shù)了,因此沒有所謂的多態(tài)。
補充說明:
1. 引用本質(zhì)上也是通過指針的解引用(即*_point)來實現(xiàn)的,可以<<參考std源碼剖析》一本書,所以引用也可以實現(xiàn)多態(tài)。
2. 即使通過 基類的指針調(diào)用基類的虛函數(shù) 或 通過子類的指針調(diào)用子類的虛函數(shù) 以及通過子類指針調(diào)用基類的虛函數(shù), 也是通過多態(tài)機制來完成的(即一步步的間接性來完成)。
3. 一個空的class的對象的大小為1個字節(jié), 編譯器之所以要這么做,是為了區(qū)別同一個類類型的不同對象!
總結(jié)
到此這篇關(guān)于c++中為什么可以通過指針或引用實現(xiàn)多態(tài)的文章就介紹到這了,更多相關(guān)c++指針或引用實現(xiàn)多態(tài)內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
C++實現(xiàn)LeetCode(557.翻轉(zhuǎn)字符串中的單詞之三)
這篇文章主要介紹了C++實現(xiàn)LeetCode(557.翻轉(zhuǎn)字符串中的單詞之三),本篇文章通過簡要的案例,講解了該項技術(shù)的了解與使用,以下就是詳細內(nèi)容,需要的朋友可以參考下2021-08-08對比C語言中g(shù)etc()函數(shù)和ungetc()函數(shù)的使用
這篇文章主要介紹了對比C語言中g(shù)etc()函數(shù)和ungetc()函數(shù)的使用,是C語言入門學習中的基礎知識,需要的朋友可以參考下2015-08-08