0

0

JVM内存模型和垃圾回收机制_JVM内存管理与自动垃圾回收关联

星降

星降

发布时间:2026-01-09 16:11:03

|

351人浏览过

|

来源于php中文网

原创

新对象默认且唯一首选分配在Eden区,仅大对象超-XX:PretenureSizeThreshold时直接入老年代;Survivor双区为复制算法服务;System.gc()仅为可忽略建议;Metaspace溢出多因类加载器泄漏。

jvm内存模型和垃圾回收机制_jvm内存管理与自动垃圾回收关联

对象分配在哪?Eden区不是“默认起点”而是“唯一首选”

新对象几乎总在 Eden 区分配,这是 JVM 分代设计的硬性约定,不是可配置的“偏好”。哪怕你把 -Xmn 设得再小,只要堆还有空闲,JVM 就不会跳过 Eden 直接往 Survivor 或老年代扔对象(大对象例外,见下条)。

  • 触发 Minor GC 的条件是 Eden 区满,不是“用了 80% 就回收”——它不看比例,只看是否还能成功分配下一个对象
  • 若对象大小超过 -XX:PretenureSizeThreshold(默认为 0,即禁用),JVM 会直接将其分配到老年代,避免在新生代反复复制造成浪费
  • 注意:字符串字面量(如 "hello")不进 Eden,它们存于堆中的运行时常量池(JDK 7+ 已移出方法区),而 new String("hello") 才会在 Eden 分配实例

为什么 Survivor 区要设两个(S0 和 S1)?不是为了“备份”,而是为复制算法腾空间

Survivor 区的存在,本质是为 复制算法 服务:每次 Minor GC 后,Eden + 当前使用的 Survivor(比如 S0)中存活的对象,被一次性复制到另一个空的 Survivor(S1),然后清空 Eden 和 S0。没有“双区”,复制就无法原子完成。

  • 比例 -XX:SurvivorRatio=8 表示 Eden : 单个 Survivor = 8:1,不是 Eden : 总 Survivor;两个 Survivor 大小相等、角色互换
  • 对象在 Survivor 中“熬过”多少次 GC 才晋升老年代,由 -XX:MaxTenuringThreshold 控制(默认 15),但实际晋升可能更早——当某次 GC 后,To Survivor 空间不足以容纳所有存活对象时,JVM 会提前把部分对象送入老年代(动态年龄判定)
  • 误配 -XX:SurvivorRatio 过大会导致 Survivor 过小,频繁触发提前晋升,加剧老年代压力;过小则浪费空间,且易因复制失败触发 Full GC

System.gc() 不是“手动触发回收”,而是向 JVM 发送一个“可忽略的建议”

调用 System.gc() 仅等价于 Runtime.getRuntime().gc(),它不保证任何行为:G1 可能完全无视,CMS 可能转为并发模式,Serial/Parallel 则大概率触发一次 Full GC——但前提是 JVM 当前没在 GC 中、没被 -XX:+DisableExplicitGC 拦截。

TemPolor
TemPolor

AI音乐生成器,一键创作免版税音乐

下载
  • 线上服务务必加 -XX:+DisableExplicitGC,否则第三方 SDK 或日志框架里一句 System.gc() 就可能引发数秒停顿
  • 显式 GC 常见于内存敏感场景(如 Android 应用退到后台),但在服务器端,它和手动调用 Thread.stop() 一样危险:破坏 JVM 自主调度节奏
  • 真正需要“及时释放”的资源(如 DirectByteBuffer),应依赖 Cleanertry-with-resources,而非寄望于 GC 回收

元空间(Metaspace)溢出 ≠ 类太多,很可能是类加载器泄漏

JDK 8+ 用本地内存实现的 Metaspace 替代了永久代,java.lang.OutOfMemoryError: Metaspace 错误出现时,第一反应不该是加 -XX:MaxMetaspaceSize,而应检查是否有自定义类加载器未被回收——每个加载器加载的类元数据都独占一块 Metaspace,且只有加载器本身被 GC 时,这些元数据才释放。

  • 典型泄漏场景:OSGi 容器、热部署框架(如 Spring Boot DevTools)、或反复 new URLClassLoader 加载同一 JAR
  • 排查命令:jstat -gc MU(Metaspace used)持续上涨;jcmd VM.native_memory summary 查 native 内存分布
  • 修复关键:确保类加载器引用链可被 GC(如清除静态 Map 缓存、关闭线程上下文类加载器绑定)
JVM 的自动垃圾回收不是“黑盒托管”,而是分代策略、算法约束与运行时反馈共同作用的结果;真正卡住性能的,往往不是 GC 算法本身,而是对象生命周期与分代假设不匹配,或是元数据管理脱离了类加载器的生命周期控制。

相关专题

更多
java
java

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

829

2023.06.15

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

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

735

2023.07.05

java自学难吗
java自学难吗

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

733

2023.07.31

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

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

396

2023.08.01

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

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

398

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中文网给大家带来了相关的视频、教程以及文章,欢迎大家前来学习阅读和下载。

16925

2023.08.03

c++主流开发框架汇总
c++主流开发框架汇总

本专题整合了c++开发框架推荐,阅读专题下面的文章了解更多详细内容。

3

2026.01.09

热门下载

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

精品课程

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

共23课时 | 2.4万人学习

C# 教程
C# 教程

共94课时 | 6.4万人学习

Java 教程
Java 教程

共578课时 | 44.1万人学习

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

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