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的更多內容!

推薦閱讀: