關(guān)于C++多重繼承下虛表結(jié)構(gòu)的問題
前言
昨天寫一個項(xiàng)目時遇到一個很詭異的現(xiàn)象 如下代碼所示:
class ParentA {
public :
int a=0x123456;
virtual void vFunParentA(){
std::cout << "vFunParentA" << std::endl;
}
};
class ParentB {
public:
int b = 0x456;
virtual void vFunParentB() {
std::cout<<"vFunParentB"<<std::endl;
int i = 0;
}
};
class Child:public ParentA,public ParentB {
int c = 0x789;
};
int main()
{
//注意這里使用void*
void* pChild = new Child;
//強(qiáng)轉(zhuǎn)
ParentB* pParentB = (ParentB*)pChild;
//強(qiáng)轉(zhuǎn)
ParentA* pParentA = (ParentA*)pChild;
pParentB->vFunParentB();
pParentA->vFunParentA();
getchar();
}
現(xiàn)在執(zhí)行后我們的預(yù)期輸出為
vFunParentB
vFunParentA
但實(shí)際輸出:
vFunParentA
vFunParentA
當(dāng)時百思不得其“姐”,最后發(fā)現(xiàn)是自己對多重繼承下的虛表內(nèi)存結(jié)構(gòu)不夠了解。
博主另一外一篇關(guān)于虛表的:
c-15 c++虛表
在多重繼承下Child內(nèi)存結(jié)構(gòu)應(yīng)該是如下結(jié)構(gòu):

由于我們使用 void * 類型強(qiáng)轉(zhuǎn)為ParentB,ParentA返回的指針地址依然是ParentA,而沒有做內(nèi)存偏移計算.如下圖所示:

當(dāng)我們不適用void*強(qiáng)轉(zhuǎn)后編譯器會幫我進(jìn)行偏移計算會迫使ParentB偏移8字節(jié),從而使pParentB 指向正確的內(nèi)存虛表。

在單一繼承下不會出現(xiàn)這類問題,我們最后看下本例中的內(nèi)存結(jié)構(gòu):




到此這篇關(guān)于關(guān)于C++ 多重繼承下虛表結(jié)構(gòu)的問題的文章就介紹到這了,更多相關(guān)C++ 多重繼承虛表內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
C++中的new/delete、構(gòu)造/析構(gòu)函數(shù)、dynamic_cast分析
這篇文章主要介紹了C++中的new/delete、構(gòu)造/析構(gòu)函數(shù)、dynamic_cast分析 本文通過實(shí)例代碼給大家介紹的非常詳細(xì),具有一定的參考借鑒價值,需要的朋友可以參考下2019-05-05
詳解C語言內(nèi)核中的鏈表與結(jié)構(gòu)體
Windows內(nèi)核中是無法使用vector容器等數(shù)據(jù)結(jié)構(gòu)的,當(dāng)我們需要保存一個結(jié)構(gòu)體數(shù)組時,就需要使用內(nèi)核中提供的專用鏈表結(jié)構(gòu)。本文分享了幾個內(nèi)核中使用鏈表存儲多個結(jié)構(gòu)體的通用案例,希望對你有所幫助2022-09-09
C語言 function recursion函數(shù)遞歸詳解
遞歸指的是在函數(shù)的定義中使用函數(shù)自身的方法,舉個例子: 從前有座山,山里有座廟,廟里有個老和尚,正在給小和尚講故事呢!故事是什么呢?"從前有座山,山里有座廟,廟里有個老和尚,正在給小和尚講故事呢!故事是什么呢?"從前有座山,山里有座廟,循環(huán)下去2021-10-10

