C++學(xué)習(xí)之虛函數(shù)表與多態(tài)詳解
概述
C++的多態(tài)在不同環(huán)境下實(shí)現(xiàn)方式可能不一樣,虛函數(shù)表是C++實(shí)現(xiàn)多態(tài)的一種方式。
問(wèn)題:
- 什么情況下C++會(huì)使用虛指針和虛函數(shù)表?
- 如果子類不新增任何虛函數(shù),也不重寫父類的虛方法,會(huì)和父類共用一張?zhí)摵瘮?shù)表么?
- 父類的構(gòu)造函數(shù)為什么不能正確的調(diào)用虛函數(shù)?
C++虛函數(shù)表指針和虛函數(shù)表
創(chuàng)建一個(gè)Base類
class Base
{
public:
int a;
int b;
};
查看Base內(nèi)存布局
1>class Base size(8):
1> +---
1> 0 | a
1> 4 | b
1> +---
為Base類添加一個(gè)虛函數(shù)
class Base
{
public:
int a;
int b;
virtual void BaseFunc1()
{
std::cout << "Call BaseFunc1 From Base" << std::endl;
};
};
此時(shí)再查看Base類的內(nèi)存布局
1>class Base size(12):
1> +---
1> 0 | {vfptr}
1> 4 | a
1> 8 | b
1> +---
1>Base::$vftable@:
1> | &Base_meta
1> | 0
1> 0 | &Base::BaseFunc1
1>Base::BaseFunc1 this adjustor: 0
Base類含有虛函數(shù)時(shí),.rodata只讀數(shù)據(jù)區(qū)會(huì)生成一個(gè)虛函數(shù)表,Base類會(huì)生成一個(gè)指向該虛函數(shù)表的指針成員變量。虛函數(shù)表存放.text代碼區(qū)函數(shù)的地址。
再為Base添加一個(gè)虛函數(shù)
class Base
{
public:
int a;
int b;
virtual void BaseFunc1()
{
std::cout << "Call BaseFunc1 From Base" << std::endl;
};
virtual void BaseFunc2()
{
std::cout << "Call BaseFunc2 From Base" << std::endl;
}
};
查看Base類的內(nèi)存分布
1>class Base size(12):
1> +---
1> 0 | {vfptr}
1> 4 | a
1> 8 | b
1> +---
1>Base::$vftable@:
1> | &Base_meta
1> | 0
1> 0 | &Base::BaseFunc1
1> 1 | &Base::BaseFunc2
1>Base::BaseFunc1 this adjustor: 0
1>Base::BaseFunc2 this adjustor: 0
Base類的虛函數(shù)表增加了一個(gè)新函數(shù)地址。
C++ 虛函數(shù)表和多態(tài)
為Base創(chuàng)建一個(gè)派生類Devire
class Derive : public Base
{
//
};
查看Derive類的內(nèi)存分布
1>class Derive size(12):
1> +---
1> 0 | +--- (base class Base)
1> 0 | | {vfptr}
1> 4 | | a
1> 8 | | b
1> | +---
1> +---
1>Derive::$vftable@:
1> | &Derive_meta
1> | 0
1> 0 | &Base::BaseFunc1
1> 1 | &Base::BaseFunc2
虛函數(shù)表的內(nèi)容和父類Base一樣
查看Base和Derive的虛函數(shù)表地址

Base和Derive并非公用一張?zhí)摵瘮?shù)表。
Derive重寫父類Base的方法
class Derive : public Base
{
public:
virtual void BaseFunc1() override
{
std::cout << "Call BaseFunc1 From Derive" << std::endl;
}
};
查看Derive類的內(nèi)存分布
1>class Derive size(12):
1> +---
1> 0 | +--- (base class Base)
1> 0 | | {vfptr}
1> 4 | | a
1> 8 | | b
1> | +---
1> +---
1>Derive::$vftable@:
1> | &Derive_meta
1> | 0
1> 0 | &Derive::BaseFunc1
1> 1 | &Base::BaseFunc2
1>Derive::BaseFunc1 this adjustor: 0
此時(shí)虛函數(shù)表的0元素被替換成了Derive::BaseFunc1的地址。
為Derive添加一個(gè)新的虛函數(shù)
class Derive : public Base
{
public:
virtual void BaseFunc1() override
{
std::cout << "Call BaseFunc1 From Derive" << std::endl;
}
virtual void DeriveFunc1()
{
std::cout << "Call DeriveFunc1" << std::endl;
}
};
再查看Derive類的內(nèi)存分布
1>class Derive size(12):
1> +---
1> 0 | +--- (base class Base)
1> 0 | | {vfptr}
1> 4 | | a
1> 8 | | b
1> | +---
1> +---
1>Derive::$vftable@:
1> | &Derive_meta
1> | 0
1> 0 | &Derive::BaseFunc1
1> 1 | &Base::BaseFunc2
1> 2 | &Derive::DeriveFunc1
1>Derive::BaseFunc1 this adjustor: 0
1>Derive::DeriveFunc1 this adjustor: 0
Derive的虛函數(shù)表添加了一個(gè)新的函數(shù)地址。
讓父類Base在構(gòu)造函數(shù)中調(diào)用虛函數(shù)BaseFunc1。
class Base
{
public:
Base()
{
BaseFunc1();
}
int a;
int b;
virtual void BaseFunc1()
{
std::cout << "Call BaseFunc1 From Base" << std::endl;
};
virtual void BaseFunc2()
{
std::cout << "Call BaseFunc2 From Base" << std::endl;
}
};
class Derive : public Base
{
public:
virtual void BaseFunc1() override
{
std::cout << "Call BaseFunc1 From Derive" << std::endl;
}
virtual void DeriveFunc1()
{
std::cout << "Call DeriveFunc1" << std::endl;
}
};
int main()
{
Derive d;
return 0;
}
輸出
Call BaseFunc1 From Base
虛函數(shù)的調(diào)用是錯(cuò)誤的。
查看Derive和Base的構(gòu)造函數(shù)的匯編代碼
Base 構(gòu)造函數(shù)匯編代碼
...
00641F4D mov dword ptr [eax],offset Base::`vftable' (0649B34h)
{
BaseFunc1();
00641F53 mov ecx,dword ptr [this]
00641F56 call Base::BaseFunc1 (0641488h)
}
00641F5B mov eax,dword ptr [this]
...
Derive 構(gòu)造函數(shù)的匯編代碼
...
0064220A mov ecx,dword ptr [this]
0064220D call Base::Base (06412B2h)
{
00642212 mov eax,dword ptr [this]
00642215 mov dword ptr [eax],offset Derive::`vftable' (0649B40h)
//
}
...
觀察匯編代碼可知,構(gòu)造Devire類的對(duì)象時(shí),當(dāng)調(diào)用父類Base的構(gòu)造函數(shù)時(shí),此時(shí)虛指針指向的虛函數(shù)表是父類Base的,只有調(diào)用Derive自己的構(gòu)造函數(shù)時(shí),虛指針被賦值為Derive的虛函數(shù)表,所以父類的構(gòu)造函數(shù)不能正確的調(diào)用虛函數(shù)。
以上就是C++學(xué)習(xí)之虛函數(shù)表與多態(tài)詳解的詳細(xì)內(nèi)容,更多關(guān)于C++虛函數(shù)表 多態(tài)的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
boost.asio框架系列之調(diào)度器io_service
這篇文章介紹了boost.asio框架系列之調(diào)度器io_service,對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2022-06-06
一起來(lái)看看C語(yǔ)言的預(yù)處理注意點(diǎn)
這篇文章主要為大家詳細(xì)介紹了C語(yǔ)言的預(yù)處理,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下,希望能夠給你帶來(lái)幫助2022-03-03
C語(yǔ)言實(shí)現(xiàn)簡(jiǎn)單飛機(jī)大戰(zhàn)
這篇文章主要為大家詳細(xì)介紹了C語(yǔ)言實(shí)現(xiàn)簡(jiǎn)單飛機(jī)大戰(zhàn),文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2020-02-02
C語(yǔ)言實(shí)現(xiàn)通訊錄系統(tǒng)課程設(shè)計(jì)
這篇文章主要為大家詳細(xì)介紹了C語(yǔ)言實(shí)現(xiàn)通訊錄系統(tǒng)課程設(shè)計(jì),文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-07-07
解決了個(gè)困擾了2天的問(wèn)題,定點(diǎn)運(yùn)算問(wèn)題
本文主要講解定點(diǎn)運(yùn)算問(wèn)題,需要的朋友可以參考一下。2016-06-06
C語(yǔ)言使用sizeof和strlen計(jì)算數(shù)組和指針大小
sizeof()一般是用來(lái)求取?變量?或者?類型?所占內(nèi)存空間的大小,strlen()是一個(gè)庫(kù)函數(shù)是專門用來(lái)計(jì)算?字符串?長(zhǎng)度的,下面我們就來(lái)看看C語(yǔ)言如何使用sizeof和strlen計(jì)算數(shù)組和指針大小吧2023-11-11

