news 2026/4/23 19:16:36

【C++】揭秘类与对象的内在机制

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
【C++】揭秘类与对象的内在机制

【C++】揭秘类与对象的内在机制

在 C++ 中,“类”和“对象”是最核心的概念,但很多人只是会用,却不知道它们在内存里到底是怎么布局的、编译器到底做了什么、虚函数表是怎么回事、this 指针藏在哪里……

这篇我们从底层视角彻底把类与对象的内存模型、构造/析构机制、虚函数机制、this 指针、多重继承下的布局等讲透。

一、类与对象在内存中的本质区别

一句话概括:

  • 是蓝图(类型定义),不占内存(只存在于编译期符号表)
  • 对象是根据蓝图实例化出来的实体,真正占用内存
classPerson{public:string name;// 8 字节(64位系统,大多数实现)intage;// 4 字节doubleheight;// 8 字节};Person p1;// p1 是对象,占 8 + 4 + 8 = 20 字节(可能有对齐,实际 24 字节)Person*p2=nullptr;// p2 是指针,只占 8 字节

结论:定义类不耗内存,创建对象才真正分配内存。

二、最简单的对象内存布局(无虚函数)

classStudent{public:string name;// 通常 24 字节(含 SSO 小字符串优化)intscore;// 4chargrade;// 1// 可能有 padding(内存对齐)};

典型 64 位系统内存布局(考虑对齐):

偏移 内容 大小 0 name 24 字节(std::string 常见实现) 24 score 4 字节 28 grade 1 字节 29~31 padding 3 字节(对齐到 4 字节) 32 (下一个成员或结束)

总大小:通常 32 字节(而非 24+4+1=29)

内存对齐规则(非常重要):

  • 每个成员按自己的对齐要求对齐(int 4 字节,double 8 字节等)
  • 整个结构体大小是对齐到最大成员的对齐边界

三、带虚函数的对象内存布局(虚表指针)

classAnimal{public:virtualvoidspeak(){cout<<"Animal\n";}virtualvoideat(){cout<<"Eat\n";}intage=0;};Animal a;

关键点:只要类中有虚函数,编译器就会在对象内存布局的最前面插入一个虚表指针(vptr),占 8 字节(64 位系统)。

内存布局变成

偏移 内容 0 vptr(指向该类的虚函数表 vtable) 8 age(4 字节 + 4 字节 padding)

虚表(vtable)是什么?

  • 每个有虚函数的(不是对象)在程序启动时(或第一次使用前)会生成一张虚函数表
  • 虚表是一个函数指针数组,按声明顺序存放虚函数地址
  • 同一个类的所有对象共享同一张虚表
Animal 的虚表(示例): 0: Animal::speak 的地址 1: Animal::eat 的地址

结论
含虚函数的类,每个对象都会多出 8 字节(vptr),指向类的虚表。

四、this 指针到底是什么?

this是一个隐式参数,每个非静态成员函数都自动带有一个隐藏的 this 指针。

voidspeak(){cout<<age;}// 编译器看到的真实样子voidspeak(Animal*constthis){cout<<this->age;}

调用方式

Animal a;a.speak();// 编译器自动传入 &a 作为 this

this 的类型

  • 在非 const 成员函数中:T* const this
  • 在 const 成员函数中:const T* const this

五、继承下的内存布局(单继承)

classAnimal{public:virtualvoidspeak(){}intage=1;};classDog:publicAnimal{public:virtualvoidspeak()override{cout<<"Woof\n";}intlegs=4;};

Dog 对象的内存布局(典型实现):

偏移 内容 0 vptr(指向 Dog 的虚表) 8 age (从 Animal 继承) 12 legs (Dog 自己的成员)

Dog 的虚表

0: Dog::speak (覆盖了 Animal::speak) 1: Animal::eat (如果有)

六、多重继承 + 虚继承下的布局(最复杂部分)

多继承时,每个基类可能有自己的 vptr,导致对象中有多个 vptr

classA{virtualvoidf();inta;};classB{virtualvoidg();intb;};classC:publicA,publicB{intc;};

C 对象可能的布局(非虚继承):

0 vptr for A 8 a 12 vptr for B 20 b 24 c

虚继承(virtual inheritance)会引入vbptr(虚基表指针),解决菱形继承问题。

虚继承布局(更复杂,包含偏移量调整):

  • 包含 vbptr(虚基表指针)
  • 虚基表记录虚基类的偏移
  • 虚函数表也会有额外的调整(thunk)

一句话总结
多继承 + 虚继承是 C++ 中最复杂的内存布局,强烈建议能不用就不用(现代设计倾向于组合优于继承)。

七、构造与析构的底层顺序

classBase{public:Base(){cout<<"Base ctor\n";}~Base(){cout<<"Base dtor\n";}};classDerived:publicBase{public:Derived(){cout<<"Derived ctor\n";}~Derived(){cout<<"Derived dtor\n";}};

构造顺序:从基类 → 派生类
析构顺序:从派生类 → 基类(与构造完全相反)

带成员变量的顺序

  1. 基类构造
  2. 成员变量按声明顺序构造(不是初始化列表顺序)
  3. 派生类构造体执行

析构完全相反

八、总结:类与对象内存机制核心要点

  1. 普通类:成员按声明顺序 + 内存对齐布局
  2. 含虚函数的类:对象开头插入 vptr(8 字节),指向类的虚表
  3. this 指针:隐式传入的指向当前对象的指针
  4. 继承:子类对象包含基类部分(前置布局)
  5. 多继承:可能有多个 vptr
  6. 虚继承:引入 vbptr + 虚基表,解决菱形问题
  7. 构造顺序:基类 → 成员 → 自身
  8. 析构顺序:自身 → 成员 → 基类

最后送你一个快速判断内存布局的口诀

  • 有虚函数? → 多 8 字节 vptr
  • 多继承? → 可能多个 vptr
  • 虚继承? → 更复杂,出现 vbptr
  • 成员变量? → 按声明顺序 + 对齐
  • 空类? → 大多数编译器占 1 字节(避免地址相同)

想继续深入哪个部分?
比如:

  • 虚函数表具体长什么样(可以用 gdb 或 dump 看)
  • 多继承/虚继承下的 this 指针调整
  • 成员函数的地址到底存哪里
  • 空基类优化(EBO)
  • 虚析构函数为什么必须有

欢迎继续问~

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/4/23 12:15:57

【嵌入式就业5】硬件体系与RTOS核心机制:从ARM架构到实时调度

【嵌入式就业5】硬件体系与RTOS核心机制:从ARM架构到实时调度 作者:石去皿 专题说明:本系列聚焦嵌入式岗位求职实战。本文为第四篇,深度剖析ARM体系架构、中断机制、RTOS调度等硬件级核心知识,结合STM32/RK3588平台工程实践,助你攻克大厂硬件相关面试难点。 一、前言:硬…

作者头像 李华
网站建设 2026/4/23 13:44:20

2026年公文报告去AIGC痕迹:工作材料这样处理

2026年公文报告去AIGC痕迹&#xff1a;工作材料这样处理 用AI写工作报告、总结汇报&#xff0c;效率是高了&#xff0c;但被领导看出来就尴尬了。 同事小王用ChatGPT写了一份年终总结&#xff0c;领导一看就说"这写的怎么像机器写的"。虽然没说什么&#xff0c;但印…

作者头像 李华
网站建设 2026/4/23 8:35:20

【Linux命令大全】010.设备管理之dumpkeys命令(实操篇)

【Linux命令大全】010.设备管理之dumpkeys命令&#xff08;实操篇&#xff09; ✨ 本文为Linux设备管理命令的全面汇总与深度优化&#xff0c;结合图标、结构化排版与实用技巧&#xff0c;专为高级用户和系统管理员打造。 (关注不迷路哈&#xff01;&#xff01;&#xff01;) …

作者头像 李华
网站建设 2026/4/23 8:32:14

【信息科学与工程学】【财务管理】第二篇 产品线财务基础篇02

产品开发财务评估全周期知识体系 编号 一级分类 二级分类 核心概念/方法/模型 关键特征/数学表述/核心输入 主要功能与应用场景 理论/法规/数据依据 I. 核心理念与价值管理​ 1 根本目标 价值导向 产品开发财务评估的核心目标​ 在产品全生命周期中&#xff0c;通过…

作者头像 李华