c++虚函数与多态是如何实现的_c++多态性原理解析

1次阅读

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

c++虚函数与多态是如何实现的_c++多态性原理解析

在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,从而确保调用的是重写后的函数。

多态的运行时绑定过程

多态的关键在于:通过基类指针或引用调用虚函数时,实际执行哪个函数由所指向对象的真实类型决定,而不是指针的声明类型。

c++虚函数与多态是如何实现的_c++多态性原理解析

CodeGeeX

智谱AI发布的AI编程辅助工具插件,可以实现自动代码生成、代码翻译、自动编写注释以及智能问答等功能

c++虚函数与多态是如何实现的_c++多态性原理解析 191

查看详情 c++虚函数与多态是如何实现的_c++多态性原理解析

这个过程发生在运行时,具体步骤如下:

  • 通过对象的 vptr 找到其所属类的 vtable
  • 在 vtable 中查找对应虚函数的地址
  • 跳转到该地址执行函数

这意味着即使 Base* 指针指向的是 Derived 对象,调用 func() 时仍会通过 Derived 的 vtable 找到 Derived::func,实现动态分发。

虚析构函数的重要性

如果基类的析构函数不是虚函数,通过基类指针删除派生类对象时,只会调用基类的析构函数,造成资源泄漏。

将析构函数声明为 virtual 后,vtable 机制确保正确调用派生类的析构函数,实现完整的对象销毁流程。

性能与内存开销

虚函数机制带来了一定的运行时开销:

  • 每个对象额外占用一个指针大小的内存(用于 vptr)
  • 每次调用虚函数需要两次寻址:先通过 vptr 找到 vtable,再查表获取函数地址

因此,只有在需要多态行为时才使用虚函数,避免不必要的开销。

基本上就这些。C++的多态不是魔法,而是编译器自动生成 vtable 和 vptr 配合完成的机制。只要理解了虚函数表的存在和调用流程,就能清晰把握多态的底层运作方式。

text=ZqhQzanResources