C++ 繼承,虛繼承(內存結構)詳解
普通的公有繼承
class test1 { public: test1(int i) :num1(i) {} private: int num1; }; class test2 : public test1 { public: test2(int i,int j) : test1(i), num2(j) { } private: int num2; }; void main() { test2 t2(1,2); }
(test2內存結構)查看內存發現父類在子類的上面
多重繼承
在原有的代碼基礎上增加瞭test3類
test3類繼承瞭 test2和test1
class test1 { public: test1(int i) :num1(i) {} private: int num1; }; class test2 { public: test2(int i) : num2(i) { } private: int num2; }; class test3 :public test2 ,public test1 { public: test3(int i, int j,int k) :test1(i), test2(j),num3(k) {} private: int num3; }; void main() { test3 t3(1, 2, 3); }
(test3內存地址 ) 依舊是父類在子類上
但是現在有兩個父類為什麼test2在test1上?
這和我們的繼承順序有關 我們先繼承瞭test2又繼承瞭test1 更換內存繼承順序 內存的情況也會有所變化
虛繼承
class test1 { public: test1(int i) :num1(i) {} private: int num1; }; class test2: virtual public test1 { public: test2(int i,int j) :test1(i), num2(j) { } private: int num2; }; void main() { test2 t2(1, 2); }
(t2的內存) 我們發現虛繼承以後父類成員數據在子類成員數據下面瞭 首地址處莫名其妙多瞭四字節
這四字節就是我們的虛基類表的地址
跟隨虛繼承表 其中存儲瞭本類距離父類對象的差值 通過差值能夠找到父類對象
我們再看這個內存0x0082fbd8是t2的首地址 0x0082fbe0是父類的位置
0x0082fbd8 – 0x0082fbe0 == 8
就是本類距離父類對象的差值
虛繼承(菱形繼承)
class test1 { public: test1(int i) :num1(i) {} private: int num1; }; class test2: virtual public test1 { public: test2(int i,int j) :test1(i), num2(j) { } private: int num2; }; class test3 :virtual public test1 { public: test3(int i, int j) :test1(i), num3(j){} private: int num3; }; class test4 :public test2, public test3 { public: test4(int i, int j, int k) :test1(i),test2(i,j), test3(i,j),num4(k) { } private: int num4; }; void main() { test4 t4(1, 2,3); }
test4的內存 我們看到 t2和t3都有自己的虛基類表地址 記錄瞭自己和父類的偏移
兩個虛基類表的內容
現在我們計算一下 到爺爺類的差值是否正確
0x00FAFD50 – 0x00fafd3c == 14
0x00FAFD50 – 0x00fafd44 == C
總結
本篇文章就到這裡瞭,希望能夠給你帶來幫助,也希望您能夠多多關註WalkonNet的更多內容!