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