C++对象模型中加入多继承
从单继承可以知道,派生类中只是扩充了基类的虚函数表。如果是多继承的话,又是如何扩充的?
1) 每个基类都有自己的虚表。
2) 子类的成员函数被放到了第一个基类的表中。
3) 内存布局中,其父类布局依次按声明顺序排列。
4) 每个基类的虚表中的print()函数都被overwrite成了子类的print ()。这样做就是为了解决不同的基类类型的指针指向同一个子类实例,而能够调用到实际的函数。
上面3个类,Derived_Mutlip_Inherit继承自Base、Base_1两个类,Derived_Mutlip_Inherit的结构如下所示:
为了验证上述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; }
输出结果如下图所示: