【C++】虚函数在不同继承方式中的对象模型

本文详细探讨了虚函数和虚拟继承的概念及其在C++中的实现机制,包括单继承、多继承、菱形继承和虚拟单继承等不同场景下对象模型的变化。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

     虚函数的调用是通过查找虚表调用的,通过虚表找到函数的地址,从而调用对应的函数,而虚表则是放在对象的前四个字节里的。

     通过基类的引用(或指针)调用虚函数时,调用基类还是派生类的函数,是根据运行时引用(或指针)的实际引用(指向)的类型确定,而调用非虚函数时,无论基类指向的是何种类型,调用的都是基类的函数。下面看看在不同继承方式下的对象模型有什么不同:

1.单继承,派生类中没有对基类函数进行重写

class Base
{
public:
	virtual void Test1() { cout << "Base::Test1()";}	
	virtual void Test2() { cout << "Base::Test2()";}
	int _b;
};
class Derived :public Base
{
public:
	virtual void Test3() { cout << "Derived::Test3()";}
	virtual void Test4() { cout << "Derived::Test4()";}
	int _d;
};
typedef void(*FuncPtr)();
void FunTest()
{
	Base b;
	b._b = 1;
	cout << "Base vfptr" << endl;
	FuncPtr* pFunc;
	for (int i = 0; i < 2; i++)
	{
		pFunc = (FuncPtr*)((int*)*(int*)&b + i);
		(*pFunc)();
		cout << ":" << (int*)*pFunc << endl;
	}

	Derived d;
	Base& base = d;
	d._b = 1;
	d._d = 2;
	cout << "Derived vfptr" << endl;
	for (int i = 0; i < 4; i++)
	{
		pFunc = (FuncPtr*)((int*)*(int*)&base + i);
		(*pFunc)();
		cout << ":" << (int*)*pFunc << endl;
	}
}
基类对象b

派生类对象d

2.单继承,派生类中对基类函数进行重写

class Base
{
public:
	virtual void Test1() { cout << "Base::Test1()";}	
	virtual void Test2() { cout << "Base::Test2()";}
	virtual void Test3() { cout << "Base::Test3()";}
	int _b;
};
class Derived :public Base
{
public:
	virtual void Test1() { cout << "Derived::Test1()";}
	virtual void Test2() { cout << "Derived::Test2()";}
	virtual void Test4() { cout << "Derived::Test4()";}
	int _d;
};
派生类对象d


由图我们可以得到对象b,d的对象模型,如下图:


可以看到:

基类中的虚表:虚函数在类中的声明次序

派生类中的虚表:1.基类中的虚表拷贝一份

                             2.检测派生类中是否对基类中的函数进行重写,用派生类中重写的函数替换相同偏移量位置的基类虚函数。

                             3.在虚表之后,添加派生类自己的虚函数

3.多继承
先继承的基类的虚表在前面,将派生类自己新增加的虚函数增加到第一张虚表中。

class Base1
{
public:
	virtual void Test1() { cout << "Base1::Test1()" << endl; }
	int _b1;
};
class Base2
{
public:
	virtual void Test2() { cout << "Base2::Test2()" << endl; }
	int _b2;
};
class Derived :public Base1,public Base2
{
public:
	virtual void Test3() { cout << "Derived::Test3()" << endl; }
	int _d;
};

4.菱形继承

class B
{
public:
	virtual void Test1() { cout << "B::Test1()" << endl; }
	virtual void Test2() { cout << "B::Test2()" << endl; }
	int _b;
};
class C1:public B
{
public:
	virtual void Test1() { cout << "C1::Test1()" << endl; }
	virtual void Test3() { cout << "C1::Test3()" << endl; }
	int _c1;
};
class C2: public B
{
public:
	virtual void Test1() { cout << "C2::Test1()" << endl; }
	virtual void Test4() { cout << "C2::Test4()" << endl; }
	int _c2;
};
class D :public C1,public C2
{
public:
	virtual void Test1() { cout << "D::Test1()" << endl; }
	virtual void Test3() { cout << "D::Test3()" << endl; }
	virtual void Test4() { cout << "D::Test4()" << endl; }
	virtual void Test5() { cout << "D::Test5()" << endl; }
	int _d;
};

5.虚拟单继承

我们知道,虚拟继承是为了解决菱形继承二义性问题而出现的,通过偏移量表格区分相同的成员。

首先看一下简单的虚拟单继承中如果类中有虚函数,是什么样的对象模型。

class Base
{
public:
	virtual void Test1() { cout << "Base::Test1()" << endl; }
	int _b;
};
class Derived :virtual public Base
{
public:
	virtual void Test1() { cout << "Derived::Test1()" << endl; }
	virtual void Test2() { cout << "Derived::Test2()" << endl; }
	int _d;
};

同虚拟继承类似,基类的数据成员在最下面,其次是基类的虚表指针,偏移量表格中的值是相对于虚表的值。需要注意的是:当派生类中没有新增的函数,即只有对基类函数的重写时,对象模型中没有第一行中的虚表指针(即派生类的虚表指针),此时,偏移量表格中的内容则是0和8。

6.菱形虚拟继承

class B
{
public:
	virtual void Test1() { cout << "B::Test1()" << endl; }
	virtual void Test2() { cout << "B::Test2()" << endl; }
	int _b;
};
class C1 :virtual public B
{
public:
	virtual void Test1() { cout << "C1::Test1()" << endl; }
	virtual void Test3() { cout << "C1::Test3()" << endl; }
	int _c1;
};
class C2 :virtual public B
{
public:
	virtual void Test1() { cout << "C2::Test1()" << endl; }
	virtual void Test4() { cout << "C2::Test4()" << endl; }
	int _c2;
};
class D :public C1, public C2
{
public:
	virtual void Test1() { cout << "D::Test1()" << endl; }
	virtual void Test3() { cout << "D::Test3()" << endl; }
	virtual void Test4() { cout << "D::Test4()" << endl; }
	virtual void Test5() { cout << "D::Test5()" << endl; }
	int _d;
};



在菱形虚拟继承的基础上,添加了虚函数。通过测试得到对象模型,由图我们可以看到,同没有虚函数的菱形虚拟继承类似,基类在对象最下方;第一个偏移量表格中的第一个值是相对于自己虚表指针的偏移量,第二个值是相对于基类虚表指针的偏移量。第二个偏移量表格是类似的;同多继承类似,派生类D中新增加的函数位于第一张虚表中。

需要注意的是:
1.虚表是所有类对象实例共用的,所以在一个类中,只有一个虚表。
2.注意区分虚函数和虚拟继承,虚函数时为了实现动态多态,虚拟继承是为了解决菱形继承二义性问题。



### C++ 虚函数与虚继承对象模型及其工作原理 #### 虚函数的工作原理 在C++中,虚函数是实现运行时多态的核心机制。当一个类定义了一个虚函数时,编译器会为该类生成一张虚函数表(vtable),这张表存储了所有虚函数的地址[^1]。对于每一个包含虚函数的类实例,都会有一个隐式的指针 `_vptr` 指向其对应的虚函数表。 以下是虚函数调用的过程: 1. 当通过基类指针或引用调用虚函数时,程序首先访问对象中的 `_vptr`。 2. 接着,通过 `_vptr` 找到虚函数表的位置,并从中获取对应虚函数的实际地址。 3. 最终执行的是派生类中覆盖的版本,而非基类中的原始定义[^2]。 这种动态绑定使得即使是在运行期也能正确调用合适的成员函数,从而支持面向对象编程中的多态特性。 ```cpp class Base { public: virtual void func() { std::cout << "Base function"; } }; class Derived : public Base { public: void func() override { std::cout << "Derived function"; } }; ``` #### 析构函数为何应声明为虚函数 为了确保删除由基类指针管理的派生类对象时能正确释放资源,通常建议将基类的析构函数设置成虚函数。这是因为如果不这样做,则仅会调用基类的析构函数而忽略掉派生部分的内容清理操作。 #### 虚继承的作用及其实现方式继承主要用于解决多重继承下的菱形问题——即防止多个父类共享同一个祖父类造成的数据冗余以及歧义现象。采用虚继承后,所有的子类都将共同拥有唯一的一份祖辈数据副本而不是各自独立复制一份出来[^3]。 具体来说,在内存布局方面,使用虚继承会使编译器引入额外的空间用于记录偏移量信息以便准确定位各个组件之间的相对位置关系;同时还会增加所谓的 `vtordisp` 字段用来辅助构造/销毁过程中的控制流调整[^3]。 下面是一个简单的例子展示如何运用虚继承: ```cpp // 定义一个虚拟基类 class VirtualBase { protected: int vb_data; public: VirtualBase(int d):vb_data(d){} virtual ~VirtualBase(){} }; // 中间层A从VB虚继承而来 class A : virtual public VirtualBase{ private: int a_specific; public: A():a_specific(0),VirtualBase(-1){} // 初始化列表顺序很重要 }; // 另外一个中间层B同样也从VB虚继承得到 class B : virtual public VirtualBase{ private: double b_value; public: B(double v=0.0):b_value(v),VirtualBase(+1){} }; // 终端类D同时继承自A和B两个分支路径上的节点 class D : public A, public B {}; ``` 以上代码片段展示了复杂的层次结构里怎样利用虚继承避免重复表示公共祖先的信息。 #### 使用方法总结 - **设计阶段**:明确哪些行为可能需要后期扩展并考虑将其标记为virtual; - **编码实践**:始终记得把顶层抽象接口类的destructor置为纯虚或者至少是普通的virtual destructor以保障安全回收整个继承体系内的全部组成部分; - **性能考量**:虽然引入了间接寻址开销但是换来灵活性往往值得付出这点代价除非特别敏感场景才需权衡利弊决定是否舍弃此功能。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值