0

0

JVM内存模型和内存结构_JVM内存模型组成与各结构功能

畫卷琴夢

畫卷琴夢

发布时间:2026-01-17 14:41:24

|

808人浏览过

|

来源于php中文网

原创

JVM内存模型(JMM)是多线程下可见性、有序性、原子性的抽象规范,而JVM内存结构是运行时数据区的物理划分(如堆、栈、方法区等);二者混淆易致调试错误或OOM。

jvm内存模型和内存结构_jvm内存模型组成与各结构功能

JVM内存模型(JMM)和JVM内存结构是两个常被混淆但完全不同的概念:前者是关于多线程下变量可见性、有序性、原子性的抽象规范,后者是运行时数据区的物理划分(如堆、、方法区等)。搞混这两者,轻则调试多线程问题绕弯路,重则误配参数导致 OOM 或 GC 频繁。

Java Memory Model(JMM)解决的是可见性与重排序问题

JMM 不是内存布局图,而是一组规则,定义了线程如何读写共享变量,以及什么情况下一个线程的修改对另一个线程“可见”。它不关心 heap 有多大,只约束 volatilesynchronizedfinal 字段和锁释放/获取的语义。

  • volatile 写操作会刷新到主内存,读操作强制从主内存加载——但不保证复合操作(如 i++)原子性
  • synchronized 块的解锁(monitorexit)会将工作内存中变量值同步回主内存;加锁(monitorenter)会清空本地工作内存,重新从主内存读取
  • 所有线程都遵守“先行发生”(happens-before)规则,例如程序次序规则、监视器锁规则、volatile 变量规则等,违反这些规则就可能看到过期值或乱序执行结果

Runtime Data Areas(内存结构)是真实分配内存的区域

这是 java -Xmx2g 真正影响的部分,由 JVM 启动时划分,对应操作系统实际申请的内存页。不同版本 JDK 的实现有差异(比如 JDK 8 的永久代 vs JDK 17 的元空间),但核心区域稳定:

  • PC Register:每个线程私有,记录当前执行字节码指令地址;唯一不会抛 OutOfMemoryError 的区域
  • Java Virtual Machine Stacks:线程私有,存储局部变量、操作数栈、动态链接、方法出口等;栈深度超限抛 StackOverflowError,总内存不足抛 OutOfMemoryError
  • Heap:所有线程共享,对象实例和数组分配在此;受 -Xms/-Xmx 控制;GC 主要作用区域
  • Method Area(JDK 8+ 为 Metaspace):存储类信息、常量、静态变量、JIT 编译代码;JDK 8 起使用本地内存,不再受 -XX:PermSize 限制,改用 -XX:MaxMetaspaceSize
  • Runtime Constant Pool:是 Method Area 的一部分,存放编译期生成的字面量(如字符串字面量)和符号引用

容易踩坑的典型场景

很多线上问题源于把 JMM 和内存结构混为一谈,进而错误归因:

网奇.NET网络商城系统
网奇.NET网络商城系统

系统优势: 1、 使用全新ASP.Net+c#和三层结构开发. 2、 可生成各类静态页面(html,htm,shtm,shtml和.aspx) 3、 管理后台风格模板自由选择,界面精美 4、 风格模板每月更新多套,还可按需定制 5、 独具的缓存技术加快网页浏览速度 6、 智能销售统计,图表分析 7、 集成国内各大统计系统 8、 多国语言支持,内置简体繁体和英语 9、 UTF-8编码,可使用于全球

下载
  • 看到 java.lang.OutOfMemoryError: Metaspace,却去调大 -Xmx——没用,得调 -XX:MaxMetaspaceSize
  • volatile 修饰 ArrayList 字段,以为能保证线程安全——其实只保证引用本身可见,add() 操作仍非原子,需用 Collections.synchronizedListCopyOnWriteArrayList
  • finalize() 方法里复活对象(如 this = new Object()),试图绕过 GC——JDK 9+ 已废弃 finalize(),且现代 GC(如 ZGC、Shenandoah)根本不保证其执行时机
  • 认为 “堆外内存不受 JVM 管理”,就随意用 ByteBuffer.allocateDirect()Unsafe.allocateMemory()——这些内存仍受 -XX:MaxDirectMemorySize 限制,超限抛 OutOfMemoryError: Direct buffer memory

验证内存结构配置是否生效的最简方式

别只看启动参数,用 jstat 实时观察实际使用情况:

jstat -gc  1000 5

输出中重点关注:

  • OGC / OC:老年代当前容量 / 最大容量,确认 -Xmx 是否生效
  • MU / MC:元空间已使用 / 最大容量,验证 -XX:MaxMetaspaceSize
  • CCSU / CCSC:压缩类空间使用量(JDK 10+),反映类加载压力

真正难的不是记住每个区域叫什么,而是当 Full GC 频繁或 Metaspace OOM 出现时,能立刻判断该查类加载器泄漏,还是该调参,而不是打开 GC 日志从头扫起。

相关专题

更多
java
java

Java是一个通用术语,用于表示Java软件及其组件,包括“Java运行时环境 (JRE)”、“Java虚拟机 (JVM)”以及“插件”。php中文网还为大家带了Java相关下载资源、相关课程以及相关文章等内容,供大家免费下载使用。

834

2023.06.15

java正则表达式语法
java正则表达式语法

java正则表达式语法是一种模式匹配工具,它非常有用,可以在处理文本和字符串时快速地查找、替换、验证和提取特定的模式和数据。本专题提供java正则表达式语法的相关文章、下载和专题,供大家免费下载体验。

739

2023.07.05

java自学难吗
java自学难吗

Java自学并不难。Java语言相对于其他一些编程语言而言,有着较为简洁和易读的语法,本专题为大家提供java自学难吗相关的文章,大家可以免费体验。

735

2023.07.31

java配置jdk环境变量
java配置jdk环境变量

Java是一种广泛使用的高级编程语言,用于开发各种类型的应用程序。为了能够在计算机上正确运行和编译Java代码,需要正确配置Java Development Kit(JDK)环境变量。php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

397

2023.08.01

java保留两位小数
java保留两位小数

Java是一种广泛应用于编程领域的高级编程语言。在Java中,保留两位小数是指在进行数值计算或输出时,限制小数部分只有两位有效数字,并将多余的位数进行四舍五入或截取。php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

399

2023.08.02

java基本数据类型
java基本数据类型

java基本数据类型有:1、byte;2、short;3、int;4、long;5、float;6、double;7、char;8、boolean。本专题为大家提供java基本数据类型的相关的文章、下载、课程内容,供大家免费下载体验。

446

2023.08.02

java有什么用
java有什么用

java可以开发应用程序、移动应用、Web应用、企业级应用、嵌入式系统等方面。本专题为大家提供java有什么用的相关的文章、下载、课程内容,供大家免费下载体验。

430

2023.08.02

java在线网站
java在线网站

Java在线网站是指提供Java编程学习、实践和交流平台的网络服务。近年来,随着Java语言在软件开发领域的广泛应用,越来越多的人对Java编程感兴趣,并希望能够通过在线网站来学习和提高自己的Java编程技能。php中文网给大家带来了相关的视频、教程以及文章,欢迎大家前来学习阅读和下载。

16926

2023.08.03

高德地图升级方法汇总
高德地图升级方法汇总

本专题整合了高德地图升级相关教程,阅读专题下面的文章了解更多详细内容。

27

2026.01.16

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Kotlin 教程
Kotlin 教程

共23课时 | 2.6万人学习

C# 教程
C# 教程

共94课时 | 6.9万人学习

Java 教程
Java 教程

共578课时 | 46.9万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号