虚函数表什么时候生成的?

news/2025/3/18 16:08:35/文章来源:https://www.cnblogs.com/Cycas/p/18779144

虚函数表(vtable)在C++中的生成时机如下:


1. 编译阶段生成

虚函数表是由编译器在编译阶段为每个包含虚函数的类生成的静态数据结构。具体来说:

  • 类定义处理:当编译器解析到类的定义中包含虚函数(包括virtual成员函数或继承自基类的虚函数)时,会为该类创建一个虚函数表。
  • 表内容确定:虚函数表中存储的是指向该类所有虚函数的指针,顺序与虚函数在类中的声明顺序一致。基类的虚函数在前,派生类新增的虚函数在后。
  • 虚函数表存储位置:虚函数表通常被放置在程序的只读数据段(如.rodata.data段),在程序加载到内存时已存在。

2. 对象构造时关联

  • 虚指针(vptr)的初始化
    每个包含虚函数的类的对象内部会隐式包含一个指向其虚函数表的指针(称为vptr)。这个vptr对象构造时被初始化:
    • 构造函数中插入代码:编译器会在类的构造函数中插入代码,将vptr指向当前类对应的虚函数表。
    • 继承链的处理:在派生类的构造函数中,vptr会依次指向基类和派生类的虚函数表(按构造顺序调整)。

3. 运行时动态绑定

  • 多态调用机制
    通过基类指针或引用调用虚函数时,程序会根据对象的vptr找到对应的虚函数表,再从表中查找目标函数的地址,实现运行时多态
  • 虚函数表不变:虚函数表本身在运行时是只读的,其内容在编译时已完全确定,不会在运行时修改。

示例代码分析

class Base {
public:virtual void func1() {}virtual void func2() {}
};class Derived : public Base {
public:void func1() override {} // 覆盖基类func1virtual void func3() {}  // 新增虚函数
};int main() {Base* obj = new Derived();obj->func1(); // 运行时通过vptr查找Derived的虚函数表,调用Derived::func1delete obj;return 0;
}
  • 编译阶段
    编译器会为BaseDerived分别生成虚函数表:
    • Base的虚函数表:[Base::func1, Base::func2]
    • Derived的虚函数表:[Derived::func1, Base::func2, Derived::func3]
  • 对象构造时
    创建Derived对象时,其vptr会被初始化为指向Derived的虚函数表。

4. 关键总结

阶段 行为
编译阶段 为每个含虚函数的类生成虚函数表,确定表中函数指针的顺序和内容。
构造对象时 初始化对象的vptr,使其指向所属类的虚函数表。
运行时 通过vptr查表实现多态调用,虚函数表内容不可变。

常见问题

Q:虚函数表会被继承吗?

  • 是的。派生类会继承基类的虚函数表,并在此基础上扩展(覆盖基类虚函数或新增虚函数)。

Q:虚函数表占用内存吗?

  • 虚函数表本身是静态数据,每个类只有一份,占用固定内存。对象的vptr占用一个指针大小的内存(通常4或8字节)。

Q:能否手动修改虚函数表?

  • 理论上可以通过指针操作修改,但属于未定义行为(UB),会导致程序崩溃或不可预测的结果。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/900983.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

初识Open3D

Step01初识Open3d 程序代码: import open3d as o3d #导入Open3d库 def main():#定义主函数box = o3d.geometry.TriangleMesh.create_box(width=3.0, height=2.0, depth=1.0)#创建一个长方体对象box.compute_vertex_normals() #计算法向量vis = o3d.visualization.Visualizer(…

20222223 实验一《Python程序设计》实验报告

20222223 2024-2025-2 《Python程序设计》实验一报告课程:《Python程序设计》 班级: 2222 姓名: 李東霖 学号: 20222223 实验教师:王志强 实验日期:2025年3月12日 必修/选修: 公选课一、实验内容 1.熟悉Python开发环境; 2.练习Python运行、调试技能;(编写书中的程序…

交换节点 - 环

树上交换节点给定一棵树,每个节点有一个权值。现在每次可以交换任意两个节点的权值,请问最少多少次交换可以使得每个节点的权值等于它的编号? 保证给出的权值是一个排列,也就是说保证一定有解。时间限制:C/C++ 1秒,其他语言2秒 空间限制:C/C++ 256M,其他语言512M输入描…

Arduino雷达使用Android移动应用程序

该系统可以检测障碍物并收集距离和角度信息,并且可以通过移动应用程序通过蓝牙进行控制。 Arduino雷达使用Android移动应用程序本项目是一个由超声波传感器和伺服电机驱动的雷达系统。该系统可以检测障碍物并收集距离和角度信息,并且可以通过移动应用程序通过蓝牙进行控制。该…

使用cursor打造智能客服demo

cursor AI它真的是非常强大。 今天讲下如何使用它,搭配deepseek api接口,来生成一个智能客服系统。这是最终的效果。首先cursor需要登录后才能使用。登录之后有两周的免费试用期。我们在窗口的右侧填写智能客服的需求。帮我实现一个网页智能客服。详细要求如下: 1.生成一个h…

【蓝牙小程序】小程序使用echart图表报错:setOption of undefined

转载自:https://developers.weixin.qq.com/community/develop/doc/0004ac054ccec0f26df7baa8756800问题:小程序使用echart图表报错 Cannot read property setOption of undefined;at api request success callback function TypeError: Cannot read property setOption of un…

前端中的Javascript

前端中的Javascript javascript定义方式内联JavaScript直接在HTML元素的事件属性中编写JavaScript代码<body><h1>Hello, World!</h1><button onclick="alert(Button clicked!)">Click Me</button> </body>内部JavaScript可以直接…

NVM:安装配置使用

一、简介 在实际的开发和学习中可能会遇到不同项目的 node 版本不同,而出现的兼容性问题。 而 nvm 就可以很好的解决这个问题,它可以在同一台机器上下管理多个 node 版本,使得程序员可以轻松地安装、卸载和切换不同的 node 版本。 在下载和配置 nvm 前,需要在控制面板中先删…

硬盘科普,M.2,PCI-E,NVMe 傻傻分不清

首先从三个层面去理解这个问题:物理接口,通道,协议 1:物理接口(相当于通讯中的电,光口) 大白话- 物理规格,像是 公路,铁路 专门跑PCI-E通道的那个物理接口:扩展性极强,可以插显卡的PCI-E X16的那个物理接口,或者插网卡,声卡的那个PCI-E X1那个物理接口,都是属于一类…

第二届长城杯ciscn半决赛awdp pwn以及应急响应wp

这次半决赛还真是状况频出,先是上午全场靶机断联了2轮,下午的应急又在坐大牢,还好是后面捋顺了逻辑做出来了,下半场干了个赛区第二,总成绩第四,这回是真燃尽了 上半场AWDP typo fix 一开始一直在改这道结果后面才发现那个prompt是真的好改,白浪费了3轮。。。 进入程序是…

LLM Assistance for Memory Safety

LLM Assistance for Memory SafetyMohammed, Nausheen, et al. "LLM Assistance for Memory Safety." 2025 IEEE/ACM 47th International Conference on Software Engineering (ICSE). IEEE Computer Society, 2024.Introduction 在软件安全的漏洞中,内存安全是主要…