当前位置: 首页 > news >正文

C++ 内存布局 - Part5: 继承关系中 构造析构与vptr的调整

这里以单继承为例,汇编采用AT&T格式,先看示例代码:

#include <iostream>class Base {
public:Base() {std::cout << "Base Constructor, this ptr: " << this << std::endl;printVptr();}virtual ~Base() {std::cout << "Base Destructor, this ptr: " << this << std::endl;printVptr();}virtual void printVptr() { std::cout << "Base::printVptr, vptr = " << *(void **)this << std::endl; }int a;int b;
};class Derived : public Base {
public:Derived() {std::cout << "Derived Constructor, this ptr: " << this << std::endl;printVptr();}~Derived() {std::cout << "Derived Destructor, this ptr: " << this << std::endl;printVptr();}void printVptr() override { std::cout << "Derived::printVptr, vptr = " << *(void **)this << std::endl; }int c;
};int main() {std::cout << "Base size: " << sizeof(Base) << std::endl;std::cout << "Derived size: " << sizeof(Derived) << std::endl;Base *obj = new Derived();std::cout << "before delete, this ptr is: " << obj << std::endl;delete obj;return 0;
}

编译运行结果:

Base size: 16
Derived size: 24
Base Constructor, this ptr: 0xa1a2c0
Base::printVptr, vptr = 0x401090
Derived Constructor, this ptr: 0xa1a2c0
Derived::printVptr, vptr = 0x401068
before delete, this ptr is: 0xa1a2c0
Derived Destructor, this ptr: 0xa1a2c0
Derived::printVptr, vptr = 0x401068
Base Destructor, this ptr: 0xa1a2c0
Base::printVptr, vptr = 0x401090

从结果中可以看到, 在执行Base部分的构造和析构时,vptr指向的是基类的虚表,执行Derived部分的构造和析构时,vptr指向的是派生类的虚表。

查看汇编代码,以Base的析构为例,其他的构造析构都是类似:

Base::~Base() [base object destructor]:pushq   %rbpmovq    %rsp, %rbpsubq    $16, %rspmovq    %rdi, -8(%rbp)             将this指针入栈movl    $vtable for Base+16, %edx  将虚表指针存入edxmovq    -8(%rbp), %rax             将this指针存入raxmovq    %rdx, (%rax)  将虚表指针存入rax指向的内存,也就是对象内存的开始部分,vptrmovl    $.LC1, %esimovl    $_ZSt4cout, %edicall    std::basic_ostream<char, std::char_traits<char> >& std::operator<< <std::char_traits<char> >(std::basic_ostream<char, std::char_traits<char> >&, char const*)movq    %rax, %rdxmovq    -8(%rbp), %raxmovq    %rax, %rsimovq    %rdx, %rdicall    std::basic_ostream<char, std::char_traits<char> >::operator<<(void const*)movl    $_ZSt4endlIcSt11char_traitsIcEERSt13basic_ostreamIT_T0_ES6_, %esimovq    %rax, %rdicall    std::basic_ostream<char, std::char_traits<char> >::operator<<(std::basic_ostream<char, std::char_traits<char> >& (*)(std::basic_ostream<char, std::char_traits<char> >&))movq    -8(%rbp), %raxmovq    %rax, %rdicall    Base::printVptr()nopleaveret

因此,整个构造析构的顺序为:

当new一个派生类对象时,首先会执行基类的构造函数,这时这个构造中的对象,其vptr指向基类的虚表,当基类部分构造完毕,继续执行派生类的构造函数时,此时对象的vptr指向派生类的虚表。

当delete这个派生类对象时,首先执行派生类的析构函数,此时对象的vptr仍然指向派生类的虚表,派生类的析构函数会继续执行基类的析构函数,此时对象的vptr会指向基类的虚表。最后派生类的析构函数会负责释放整个对象内存: call    operator delete(void*, unsigned long)

http://www.lryc.cn/news/460048.html

相关文章:

  • BUG-AttributeError: ‘EnforcedForest‘ object has no attribute ‘node‘
  • Spring Boot 3 配置 Redis 兼容单例和集群
  • unsat钱包签名算法解析
  • mysql删除唯一索引
  • 学习之面试题:偏函数
  • 面试技术点
  • 基础sql
  • Jenkins整合Docker实现CICD自动化部署(若依项目)
  • kali chrome 安装 hackbar
  • 一文了解 Linux 系统的文件权限管理
  • Spark:DataFrame介绍及使用
  • Linux系统:本机(物理主机)访问不了虚拟机中的apache服务问题的解决方案
  • 望繁信科技成功签约国显科技 流程挖掘助力制造业智造未来
  • 枚举在Java体系中的作用
  • 『气泡水』Web官网 案例赏析
  • 【前端】制作一个简单的网页(2)
  • OpenAI Canvas:提升编程与写作效率的全新工作界面
  • 将SpringBoot的Maven项目打成jar包和war包
  • 【Iceberg分析】Spark与Iceberg集成之常用存储过程
  • [旧日谈]关于Qt的刷新事件频率,以及我们在Qt的框架上做实时的绘制操作时我们该关心什么。
  • 云上考场小程序+ssm论文源码调试讲解
  • 城域网——IP城域网、城域以太网、光城域网
  • 华为Eth-trunk链路聚合加入到E-trunk实现跨设备的链路聚合
  • 【网络安全】JSONP劫持原理及攻击实战
  • VR全景摄影的拍摄和编辑软件推荐
  • linux:使用sar诊断问题
  • CUDA编程技巧(不断搜集更新)
  • 云计算(第二阶段):mysql后的shell
  • Debian12离线部署Mysql全网最详细教程
  • 文本生成视频技术:艺术与科学的交汇点