虚函数是C++中实现运行时多态的核心机制,通过在基类中使用virtual声明函数,允许派生类重写该函数;当通过基类指针或引用调用虚函数时,程序会根据对象实际类型动态绑定到对应版本的函数。示例中Animal类的speak()被声明为虚函数,Dog和Cat类重写它;使用Animal*指针指向Dog或Cat对象并调用speak()时,会正确调用各自实现,输出“Dog barks”或“Cat meows”。该机制依赖虚函数表(vtable)和对象内部的虚指针(vptr),每个含虚函数的类都有一个vtable存储虚函数地址,对象通过vptr在运行时查找并调用正确的函数。注意事项包括:基类析构函数应声明为虚函数以防止资源泄漏,建议使用override关键字确保正确重写,避免对静态函数或构造函数使用虚函数,且需注意虚函数带来的间接调用性能开销。掌握虚函数与多态的关键在于理解“基类指针 + 虚函数 + 派生类重写 = 运行时决定调用函数”的机制。

在C++中,虚函数是实现多态的核心机制。通过虚函数,程序可以在运行时根据对象的实际类型调用对应的函数版本,而不是根据指针或引用的声明类型。这种行为称为动态绑定或运行时多态。
什么是虚函数?
虚函数是在基类中使用virtual关键字声明的成员函数,它允许派生类重写该函数。当通过基类指针或引用调用该函数时,会自动调用实际对象类型的函数版本。
示例:
class Animal {
public:
virtual void speak() {
std::cout << "Animal speaks\n";
}
};
class Dog : public Animal {
public:
void speak() override {
std::cout << "Dog barks\n";
}
};
class Cat : public Animal {
public:
void speak() override {
std::cout << "Cat meows\n";
}
};
上面代码中,speak()在基类Animal中被声明为虚函数,Dog和Cat分别重写了它。
立即学习“C++免费学习笔记(深入)”;
如何实现运行时多态?
多态的关键在于使用基类指针或引用指向派生类对象,并调用虚函数。
示例代码:
Animal* ptr;Dog dog; Cat cat;
ptr = &dog; ptr->speak(); // 输出: Dog barks
ptr = &cat; ptr->speak(); // 输出: Cat meows
虽然ptr是Animal*类型,但调用speak()时,程序会根据它实际指向的对象类型来决定调用哪个版本。这就是多态的体现。
虚函数的工作机制:虚函数表(vtable)
C++编译器通常使用虚函数表(vtable)来实现虚函数的动态调用。
- 每个包含虚函数的类都有一个隐藏的虚函数表,表中存储了该类所有虚函数的地址。
- 每个对象内部会有一个指向其类虚函数表的指针(vptr)。
- 当调用虚函数时,程序通过对象的vptr找到vtable,再查表获取正确的函数地址。
这个过程在运行时完成,因此实现了动态绑定。
注意事项与最佳实践
- 基类中的析构函数如果可能被继承,应声明为虚函数,否则删除派生类对象时可能导致资源泄漏。
- 使用override关键字明确表示函数意在重写基类虚函数,有助于编译器检查错误。
- 虚函数有一定性能开销(间接寻址),不必要时不要滥用。
- 静态函数、构造函数不能是虚函数。
基本上就这些。掌握虚函数和多态,是理解C++面向对象高级特性的关键一步。核心就是:基类指针 + 虚函数 + 派生类重写 = 运行时决定调用哪个函数。不复杂但容易忽略细节。











