C++对象模型5--多继承下的对象模型

C++对象模型中加入多继承


 

从单继承可以知道,派生类中只是扩充了基类的虚函数表。如果是多继承的话,又是如何扩充的?

1)        每个基类都有自己的虚表。

2)        子类的成员函数被放到了第一个基类的表中。

3)        内存布局中,其父类布局依次按声明顺序排列。

4)        每个基类的虚表中的print()函数都被overwrite成了子类的print ()。这样做就是为了解决不同的基类类型的指针指向同一个子类实例,而能够调用到实际的函数。

clip_image023[3]

上面3个类,Derived_Mutlip_Inherit继承自BaseBase_1两个类,Derived_Mutlip_Inherit的结构如下所示:

clip_image025[3]

为了验证上述C++对象模型,我们编写如下测试代码。

void test_multip_inherit()
{
    Derived_Mutlip_Inherit dmi(3333);
    cout << "对象dmi的起始内存地址:		" << &dmi << endl;
    cout << "虚函数表_vptr_Base地址:	" << (int*)(&dmi) << endl;
    cout << "_vptr_Base — 第1个函数地址:	" << (int*)*(int*)(&dmi) << "	即析构函数地址" << endl;
    cout << "_vptr_Base — 第2个函数地址:	" << ((int*)*(int*)(&dmi) + 1) << "	";
    typedef void(*Fun)(void);
    Fun pFun = (Fun)*(((int*)*(int*)(&dmi)) + 1);
    pFun();
    cout << endl;
    cout << "_vptr_Base — 第3个函数地址:	" << ((int*)*(int*)(&dmi) + 2) << "	";
    pFun = (Fun)*(((int*)*(int*)(&dmi)) + 2);
    pFun();
    cout << endl;
    cout << "_vptr_Base — 第4个函数地址:	" << *((int*)*(int*)(&dmi) + 3) << "【结束】	";
    cout << endl;
    cout << "推测数据成员iBase地址:		" << ((int*)(&dmi) +1) << "	通过地址取得的值:" << *((int*)(&dmi) +1) << endl;
 
 
    SetConsoleTextAttribute(GetStdHandle(STD_OUTPUT_HANDLE), FOREGROUND_INTENSITY | FOREGROUND_GREEN);
    cout << "++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++" << endl;
    SetConsoleTextAttribute(GetStdHandle(STD_OUTPUT_HANDLE), FOREGROUND_INTENSITY | FOREGROUND_RED);
    cout << "虚函数表_vptr_Base1地址:	" << ((int*)(&dmi) +2) << endl;
    cout << "_vptr_Base1 — 第1个函数地址:	" << (int*)*((int*)(&dmi) +2) << "	即析构函数地址" << endl;
    cout << "_vptr_Base1 — 第2个函数地址:	" << ((int*)*((int*)(&dmi) +2) + 1) << "	";
    typedef void(*Fun)(void);
    pFun = (Fun)*((int*)*((int*)(&dmi) +2) + 1);
    pFun();
    cout << endl;
    cout << "_vptr_Base1 — 第3个函数地址:	" << *((int*)*(int*)((int*)(&dmi) +2) + 2) << "【结束】	";
    cout << endl;  
    cout << "推测数据成员iBase1地址:	" << ((int*)(&dmi) +3) << "	通过地址取得的值:" << *((int*)(&dmi) +3) << endl;
    SetConsoleTextAttribute(GetStdHandle(STD_OUTPUT_HANDLE), FOREGROUND_INTENSITY | FOREGROUND_GREEN);
    cout << "++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++" << endl;
    SetConsoleTextAttribute(GetStdHandle(STD_OUTPUT_HANDLE), FOREGROUND_INTENSITY | FOREGROUND_RED);
    cout << "推测数据成员iDerived地址:	" << ((int*)(&dmi) +4) << "	通过地址取得的值:" << *((int*)(&dmi) +4) << endl;
}

  输出结果如下图所示:

clip_image027[3]

原文地址:https://www.cnblogs.com/stemon/p/4673602.html