news 2026/6/9 21:28:30

【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/6/6 19:49:55

‌自动化伦理框架:防止AI过度依赖的设计原则

在当前软件测试领域&#xff0c;AI驱动的测试用例生成、缺陷预测、日志分析与自动化脚本修复工具已广泛渗透至CI/CD流水线。然而&#xff0c;技术便利背后潜藏的‌伦理风险‌——即测试人员对AI系统的‌系统性依赖‌——正悄然削弱测试团队的专业判断力、掩盖测试盲区&#xff…

作者头像 李华
网站建设 2026/6/9 21:24:41

DeepSeek总结的SNKV — 无查询处理器的 SQLite 键值存储

原文地址&#xff1a;https://github.com/hash-anu/snkv SNKV — 无查询处理器的 SQLite 键值存储 概述 SNKV 是一个轻量级、高性能、ACID 兼容的键值存储&#xff0c;直接构建在 SQLite B‑Tree 层 之上。 与通过 SQL 查询使用 SQLite 不同&#xff0c;SNKV 绕过了整个 SQ…

作者头像 李华
网站建设 2026/6/8 15:23:53

linux编译报错

/usr/bin/ld: 找不到 -lncursesw $:~/workspace/kernel-driver/linux-xlnx-xlnx_rebase_v5.4_2020.2$ make menuconfig HOSTLD scripts/kconfig/mconf /usr/bin/ld: 找不到 -lncursesw# 解决 $:sudo ln -s /usr/lib/x86_64-linux-gnu/libncurses.so /usr/lib/x86_64-linux-gn…

作者头像 李华
网站建设 2026/6/8 14:17:02

基于Spring Boot的人格测试网站

&#x1f345; 作者主页&#xff1a;Selina .a &#x1f345; 简介&#xff1a;Java领域优质创作者&#x1f3c6;、专注于Java技术领域和学生毕业项目实战,高校老师/讲师/同行交流合作。 主要内容&#xff1a;SpringBoot、Vue、SSM、HLMT、Jsp、PHP、Nodejs、Python、爬虫、数据…

作者头像 李华