c++多态通过虚函数表(vtable)和虚函数指针(vptr)实现;2. 每个含虚函数的类对象包含指向vtable的vptr;3. 调用虚函数时,通过vptr找到vtable,再查表确定函数地址并执行;4. 基类指针调用虚函数可动态绑定到派生类函数;5. 虚析构函数确保正确析构派生类对象;6. 该机制带来每对象一个指针的内存开销及两次寻址的运行时开销。

在C++中,虚函数和多态的实现依赖于编译器内部的一套机制,核心是虚函数表(vtable)和虚函数指针(vptr)。理解这些底层原理,有助于掌握多态是如何在运行时动态绑定函数调用的。
虚函数表与虚函数指针
当一个类声明了虚函数(或继承了虚函数),编译器会为该类生成一个虚函数表,简称vtable。这个表是一个函数指针数组,存储了该类所有虚函数的实际地址。
每个包含虚函数的类的对象,都会在内存中隐式包含一个指向其类vtable的指针,称为vptr。这个指针通常位于对象内存布局的最开始位置。
例如:
假设我们有基类 Base 和派生类 Derived:
立即学习“C++免费学习笔记(深入)”;
class Base { virtual void func() { cout <br><code>};class Derived : public Base { void func() override { cout <br><code>};
编译器会为 Base 和 Derived 分别创建 vtable:
- Base 的 vtable 中 func 指向 Base::func
- Derived 的 vtable 中 func 指向 Derived::func
当创建 Derived 对象时,其内部的 vptr 会指向 Derived 类的 vtable,从而确保调用的是重写后的函数。
多态的运行时绑定过程
多态的关键在于:通过基类指针或引用调用虚函数时,实际执行哪个函数由所指向对象的真实类型决定,而不是指针的声明类型。
这个过程发生在运行时,具体步骤如下:
- 通过对象的 vptr 找到其所属类的 vtable
- 在 vtable 中查找对应虚函数的地址
- 跳转到该地址执行函数
这意味着即使 Base* 指针指向的是 Derived 对象,调用 func() 时仍会通过 Derived 的 vtable 找到 Derived::func,实现动态分发。
虚析构函数的重要性
如果基类的析构函数不是虚函数,通过基类指针删除派生类对象时,只会调用基类的析构函数,造成资源泄漏。
将析构函数声明为 virtual 后,vtable 机制确保正确调用派生类的析构函数,实现完整的对象销毁流程。
性能与内存开销
虚函数机制带来了一定的运行时开销:
- 每个对象额外占用一个指针大小的内存(用于 vptr)
- 每次调用虚函数需要两次寻址:先通过 vptr 找到 vtable,再查表获取函数地址
因此,只有在需要多态行为时才使用虚函数,避免不必要的开销。
基本上就这些。C++的多态不是魔法,而是编译器自动生成 vtable 和 vptr 配合完成的机制。只要理解了虚函数表的存在和调用流程,就能清晰把握多态的底层运作方式。