多态通过虚函数和vtable实现,基类指针调用时经vptr查找派生类函数地址,实现运行时绑定,支持接口统一与行为扩展。

多态是C++面向对象编程的核心特性之一,它允许通过基类指针或引用调用派生类的函数。实现这一机制的关键在于虚函数和动态绑定。理解其底层原理有助于写出更高效、可扩展的代码。
虚函数与虚函数表(vtable)
当一个类中声明了虚函数,编译器会为该类生成一个虚函数表(vtable)。这个表是一个函数指针数组,存储了该类所有虚函数的实际地址。每个含有虚函数的类都有自己的vtable。
同时,编译器会在对象中隐式添加一个指向其类vtable的指针,称为vptr。这个指针在构造对象时被初始化,指向对应类的vtable。
例如:
立即学习“C++免费学习笔记(深入)”;
class Base {
public:
virtual void func() { cout << "Base::func" << endl; }
};
class Derived : public Base {
public:
void func() override { cout << "Derived::func" << endl; }
};
在内存中,Derived对象的vptr会指向Derived类的vtable,其中的func()条目指向Derived::func。这样即使使用Base*指针指向Derived对象,也能正确调用派生类函数。
动态绑定的执行过程
动态绑定发生在运行时,具体流程如下:
- 通过基类指针或引用调用虚函数时,程序首先访问对象的vptr
- 通过vptr找到对应的vtable
- 在vtable中查找该虚函数的地址
- 跳转到实际函数地址执行
这种间接调用方式虽然比直接调用稍慢,但实现了接口统一和行为多态。非虚函数则在编译期就确定调用目标,属于静态绑定。
纯虚函数与抽象类
C++还支持纯虚函数:
virtual void func() = 0;
含有纯虚函数的类称为抽象类,不能实例化。派生类必须实现所有纯虚函数,否则仍是抽象类。这为接口设计提供了有力支持,类似于其他语言中的interface概念。
基本上就这些。虚函数机制虽由编译器自动管理,但了解vtable和vptr的工作方式,能帮助我们避免常见陷阱,比如在构造函数或析构函数中调用虚函数时,动态绑定可能不会按预期工作。掌握这些细节,才能真正用好多态。不复杂但容易忽略。










