淺談C++對象的內存分布和虛函數(shù)表
c++中一個類中無非有四種成員:靜態(tài)數(shù)據成員和非靜態(tài)數(shù)據成員,靜態(tài)函數(shù)和非靜態(tài)函數(shù)。
1、非靜態(tài)數(shù)據成員被放在每一個對象體內作為對象專有的數(shù)據成員。
2、靜態(tài)數(shù)據成員被提取出來放在程序的靜態(tài)數(shù)據區(qū)內,為該類所有對象共享,因此只存在一份。
3、靜態(tài)和非靜態(tài)成員函數(shù)最終都被提取出來放在程序的代碼段中并為該類所有對象共享,因此每一個成員函數(shù)也只能存在一份代碼實體。在c++中類的成員函數(shù)都是保存在靜態(tài)存儲區(qū)中的 ,那靜態(tài)函數(shù)也是保存在靜態(tài)存儲區(qū)中的,他們都是在類中保存同一個憊份。
因此,構成對象本身的只有數(shù)據,任何成員函數(shù)都不隸屬于任何一個對象,非靜態(tài)成員函數(shù)與對象的關系就是綁定,綁定的中介就是this指針。成員函數(shù)為該類所有對象共享,不僅是處于簡化語言實現(xiàn)、節(jié)省存儲的目的,而且是為了使同類對象有一致的行為。同類對象的行為雖然一致,但是操作不同的數(shù)據成員。
#include"iostream.h" class A { private: int x; int y; public: void sety() { y=5; } void print() { cout<<"x="<<x<<endl<<"y="<<y<<endl; } }; void main() { A a; a.sety(); a.print(); int *p=(int *)&a;//對象a的內存模型里面,只有非靜態(tài)數(shù)據成員,所以是合理的。 *p=6; a.print(); }
4、單繼承的對象的內存布局,第一個為虛函數(shù)表指針vtbl,其后為成員且先基類后子類,虛函數(shù)表里包含了所有的虛函數(shù)的地址,以NULL結束。虛函數(shù)如果子類有重寫,就由子類的重新的代替。
<一> 上述類圖的解釋:B的v2覆蓋了A的v2,C的v1覆蓋了A的v1,C的v3覆蓋了B的v3.
注意:發(fā)生覆蓋的特征有:
1) 不同的范圍(分別位于派生類與基類);
2) 函數(shù)名字相同;
3) 參數(shù)相同;
4) 基類函數(shù)必須有virtual關鍵字。
成員函數(shù)被重載的特征
(1)相同的范圍(在同一個類中);
(2)函數(shù)名字相同;
(3)參數(shù)不同;
(4)virtual 關鍵字可有可無。
“隱藏”是指派生類的函數(shù)屏蔽了與其同名的基類函數(shù),規(guī)則如下
(1)如果派生類的函數(shù)與基類的函數(shù)同名,但是參數(shù)不同。此時,不論有無virtual關鍵字,基類的函數(shù)將被隱藏(注意別與重載混淆)。
(2)如果派生類的函數(shù)與基類的函數(shù)同名,并且參數(shù)也相同,但是基類函數(shù)沒有virtual 關鍵字。此時,基類的函數(shù)被隱藏(注意別與覆蓋混淆)
<二> 類C的對象的內存模型為:
單繼承的對象的布局,第一個為虛函數(shù)表指針vtbl,其后為成員且先基類后子類,虛函數(shù)表里包含了所有的虛函數(shù)的地址,以NULL結束。虛函數(shù)如果子類有重寫,就由子類的重新的代替。
5、與單繼承相同的是所有的虛函數(shù)都包含在虛函數(shù)表中,所不同的多重繼承有多個虛函數(shù)表,當子類對父類的虛函數(shù)有重寫時,子類的函數(shù)替換父類的函數(shù)在對應的虛函數(shù)位置,當子類有新的虛函數(shù)時,這些虛函數(shù)被加在第一個虛函數(shù)表的后面。
<一> 對類圖的解釋:D的v3覆蓋了B3的v3,另外D有一個新的虛函數(shù)vD。
<二> class D 的內存模型
總結:與單繼承相同的是所有的虛函數(shù)都包含在虛函數(shù)表中,所不同的多重繼承有多個虛函數(shù)表,當子類對父類的虛函數(shù)有重寫時,子類的函數(shù)覆蓋父類的函數(shù)在對應的虛函數(shù)位置,當子類有新的虛函數(shù)時,這些虛函數(shù)被加在第一個虛函數(shù)表的后面
6. 虛繼承,使公共的基類在子類中只有一份,我們看到虛繼承在多重繼承的基礎上多了vbtable來存儲到公共基類的偏移。
<一>類圖的解釋:D2覆蓋了B的vB,GD覆蓋了D1的vD1同時也覆蓋B的vB,GD也有自己的虛函數(shù)vGD。
<二> class GD的內存模型
總結:虛繼承,使公共的基類在子類中只有一份,我們看到虛繼承在多重繼承的基礎上多了vbtable來存儲到公共基類的偏移
以上就是小編為大家?guī)淼臏\談C++對象的內存分布和虛函數(shù)表全部內容了,希望大家多多支持腳本之家~