C++中的虚表和虚表指针的原理和示例

一、基本概念

1. 什么是虚函数(virtual function)?

虚函数是用 virtual 关键字修饰的成员函数,支持运行时多态(dynamic polymorphism)。通过基类指针或引用调用派生类重写的函数。

cpp 复制代码
class Base {
public:
    virtual void speak() { cout << "Base speaking" << endl; }
};

只要类中存在虚函数,编译器就会为该类生成虚表。


2. 什么是虚表(vtable)?

  • vtable 是一个函数指针数组,用于保存该类的所有虚函数的地址。
  • 每个有虚函数的类有一张虚表
  • 如果派生类重写了虚函数,虚表中对应的函数指针将被替换为派生类版本。

3. 什么是虚表指针(vptr)?

  • 每个对象中都有一个隐藏成员变量 vptr(虚表指针),指向该对象所属类的虚表。
  • 对象创建时,构造函数会自动设置 vptr 的值。
  • 调用虚函数时,程序会通过 vptr 找到虚表,再通过表中函数指针找到目标函数,最终调用。

二、图解原理(简化示意)

text 复制代码
对象结构:          虚表结构(函数地址表):

[对象内存]
+---------+        +------------------+
|  vptr   | -----> | &Derived::speak()|
+---------+        +------------------+

三、示例代码与解析

示例:基类与派生类使用虚函数

cpp 复制代码
#include <iostream>
using namespace std;

class Base {
public:
    virtual void speak() {
        cout << "Base::speak()" << endl;
    }
};

class Derived : public Base {
public:
    void speak() override {
        cout << "Derived::speak()" << endl;
    }
};

int main() {
    Base* p = new Derived();  // p 的 vptr 指向 Derived 的虚表
    p->speak();               // 动态绑定,输出 Derived::speak()

    delete p;
    return 0;
}

执行过程(底层原理):

  1. 创建 Derived 对象,构造函数自动设置 vptr,指向 Derived 的虚表;
  2. 虚表中 speak() 的指针是 &Derived::speak()
  3. 调用 p->speak(),程序先通过 vptr 找到虚表,再调用函数指针,最终运行 Derived::speak()

四、进一步理解:虚表模拟(伪代码)

下面是 C++ 编译器幕后自动完成的模拟行为:

cpp 复制代码
class Base {
    void** vptr;  // 隐藏成员:虚表指针
    static void* vtable_Base[] = { &Base::speak };

public:
    virtual void speak();
};

class Derived : public Base {
    static void* vtable_Derived[] = { &Derived::speak };

public:
    void speak() override;
};

你不会在代码中显式看到 vptrvtable,它们是编译器隐藏实现的。


五、相关细节注意

情况 说明
类没有虚函数 不生成 vtable,不支持运行时多态
虚函数未被重写 虚表中仍然是基类函数指针
多重继承 每个父类一套虚表和一个 vptr
析构函数建议设为 virtual 防止只析构父类对象导致内存泄漏
构造函数中调用虚函数 不会发生多态,vptr 还没完全初始化

六、下节预告

1.多重继承下的虚表结构

2.析构函数不设为virtual导致内存泄漏示例

相关推荐
Dream achiever21 分钟前
6.WPF控件---Label
开发语言·wpf
360智汇云1 小时前
k8s共享存储fuse-client三种运行方案对比
java·linux·开发语言
先知后行。1 小时前
QT音视频
开发语言·qt·音视频
二向箔reverse1 小时前
人脸特征可视化进阶:用 dlib+OpenCV 绘制面部轮廓与器官凸包
开发语言·python
寒月霜华1 小时前
java-File
java·开发语言
钱彬 (Qian Bin)2 小时前
企业级实战:构建基于Qt、C++与YOLOv8的模块化工业视觉检测系统(基于QML)
c++·qt·yolo·qml·工业质检·qt 5.15.2
会开花的二叉树2 小时前
上手 cpp-httplib:轻量级 C++ HTTP 库的安装与实战指南
开发语言·c++·http
周杰伦fans2 小时前
C# 集合框架完全指南:从IEnumerable到ObservableCollection的深度解析
开发语言·c#
秦禹辰2 小时前
开源多场景问答社区论坛Apache Answer本地部署并发布至公网使用
开发语言·后端·golang
Su^!-苏释州2 小时前
Windows配置C/C++环境:MinGW+Vscode
c语言·c++·windows·vscode·大一新生学c语言