zoukankan      html  css  js  c++  java
  • 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]

  • 相关阅读:
    软工1816 · 作业(十一)事后诸葛亮
    软工1816·Alpha冲刺(10/10)
    软工1816 · Alpha冲刺(9/10)
    软工1816 · Alpha冲刺(8/10)
    软工1816 · Alpha冲刺(7/10)
    软工1816 · Alpha冲刺(6/10)
    软工1816 · Alpha冲刺(5/10)
    软工1816 · 作业(九)团队现场编程实战
    团队项目评测
    Beta冲刺前准备
  • 原文地址:https://www.cnblogs.com/stemon/p/4673602.html
Copyright © 2011-2022 走看看