虚函数表(vtable)和虚指针(vptr)是C++实现多态的核心机制。编译器为含虚函数的类生成vtable,存储虚函数地址;对象中隐式包含vptr,指向所属类的vtable,通常置于对象起始位置。继承时,派生类修改vtable中被重写的函数条目并添加新虚函数。通过基类指针调用虚函数时,程序通过vptr查找vtable,确定实际调用的函数,实现动态绑定。多重继承可能导致多个vptr,增加对象大小和调用开销。虚函数带来运行时多态能力,但也引入查表开销,无法内联,影响性能。了解vtable和vptr有助于理解多态底层机制及其代价。

虚函数表(vtable)和虚指针(vptr)是C++实现多态的核心机制。理解它们的工作原理,有助于深入掌握对象在运行时如何动态调用函数。
虚函数表与vptr的基本概念
当一个类声明了虚函数,编译器会为该类生成一张虚函数表(vtable)。这张表本质上是一个函数指针数组,存储了该类所有虚函数的实际地址。每个包含虚函数的类都有自己的vtable。
同时,编译器会在对象中插入一个隐式成员——vptr(虚指针),它指向所属类的vtable。这个指针通常在构造函数中由编译器自动初始化。
对象内存布局中的vptr位置
对于含有虚函数的类,其对象的内存布局中会多出一个指针成员(vptr)。多数编译器将vptr放在对象的起始位置。
立即学习“C++免费学习笔记(深入)”;
例如:
class Base {public:
virtual void func1() { }
virtual void func2() { }
};
Base b;
此时对象b的内存结构大致如下:
- 前8字节(64位系统):vptr,指向Base类的vtable
- 后续空间:Base的成员变量(如有)
vtable中按声明顺序存放func1和func2的地址。
继承与多态中的vtable行为
派生类会继承基类的vtable结构,并根据重写情况进行修改。
比如:
class Derived : public Base {public:
void func1() override { } // 重写
virtual void func3() { } // 新增虚函数
};
Derived类的vtable内容为:
- func1:指向Derived::func1(覆盖了Base的版本)
- func2:仍指向Base::func2(未重写)
- func3:新增项,指向Derived::func3
当通过基类指针调用虚函数时:
Base* ptr = new Derived();ptr->func1(); // 实际调用 Derived::func1()
执行过程是:
- 取出ptr指向对象的vptr
- 通过vptr找到vtable
- 查表获取func1对应的函数地址
- 跳转执行
多重继承与虚函数表的复杂性
在多重继承中,对象可能包含多个vptr,分别指向不同基类的vtable。这会导致对象尺寸增大,并在类型转换时涉及指针调整。
使用虚继承时,还会引入额外的间接层来解决菱形继承问题,vptr的管理也更复杂,但核心仍是通过查表实现动态绑定。
基本上就这些。vtable和vptr是编译器自动生成的机制,开发者无需手动干预,但了解其存在能更好理解多态的开销与限制,比如虚函数调用比普通函数慢,因为多了查表步骤;而内联也无法应用于虚函数的动态调用路径。不复杂但容易忽略。










