Java虚拟机之垃圾收集(一)

目录

一、如何判定对象“生死”?

1. 引用计数算法(理论参考)

2. 可达性分析算法(JVM 实际使用)

3. 对象的“缓刑”机制

二、引用类型与回收策略

三、何时触发垃圾回收?

1. 分代回收策略

2. 手动触发与注意事项

四、垃圾回收算法与实现

1. 基础算法对比

2. 分代收集理论

3. 新生代回收:Apple式复制算法

五、主流垃圾收集器详解

1. CMS 收集器(低停顿优先)

2. G1 收集器(平衡吞吐与延迟)

3. 收集器对比

六、调优建议与工具推荐

1. 参数配置示例

2. 常见问题排查

3. 工具推荐

七、总结


一、如何判定对象“生死”?

垃圾收集(GC)的核心是识别无用对象。JVM 通过两种算法判断对象是否存活:

1. 引用计数算法(理论参考)

  • 原理
    每个对象维护一个引用计数器,被引用时计数器 +1,引用失效时 -1。计数器为 0 时判定为可回收。

  • 缺点
    无法解决循环引用问题(如对象 A 引用 B,B 也引用 A)。

  • Java 未采用:主流 JVM 均使用 可达性分析算法

2. 可达性分析算法(JVM 实际使用)

  • 原理
    从 GC Roots 出发,遍历对象引用链。若对象无法被 GC Roots 关联,则判定为可回收。

  • GC Roots 对象类型

    • 虚拟机栈中的局部变量(如方法参数、局部变量)。

    • 方法区中静态变量引用的对象。

    • 方法区中常量引用的对象(如字符串常量池)。

    • 本地方法栈中 JNI 引用的对象(Native 方法)。

    • 同步锁持有的对象(synchronized 锁对象)。

    • Java 虚拟机内部对象(如系统类加载器、异常对象)。

3. 对象的“缓刑”机制

  • finalize() 方法
    若对象重写 finalize() 且未被调用过,JVM 会将其放入 F-Queue,由 Finalizer 线程触发该方法。

  • 逃脱机会
    在 finalize() 中重新建立与 GC Roots 的引用链,可避免被回收(仅一次)。

public class RescueObject {
    public static RescueObject hook;

    @Override
    protected void finalize() throws Throwable {
        super.finalize();
        hook = this; // 在 finalize 中自我拯救
    }
}

二、引用类型与回收策略

Java 提供 四种引用类型,控制对象生命周期与回收优先级:

引用类型特点回收时机典型场景
强引用Object obj = new Object(),默认引用类型对象不可达时回收普通对象创建
软引用SoftReference<Object> ref = new SoftReference<>(obj)内存不足时回收(OOM 前触发)缓存(如图片缓存)
弱引用WeakReference<Object> ref = new WeakReference<>(obj)下一次 GC 时回收临时缓存(如 WeakHashMap)
虚引用PhantomReference<Object> ref = new PhantomReference<>(obj, queue)随时可能回收,需配合 ReferenceQueue 使用堆外内存回收监听(如 DirectByteBuffer)

三、何时触发垃圾回收?

GC 触发时机由 内存区域分配策略 和 JVM 配置参数 共同决定:

1. 分代回收策略

区域GC 类型触发条件
新生代Minor GCEden 区空间不足
老年代Major GC老年代空间不足(通常伴随 Full GC)
整堆Full GC方法区不足、老年代空间不足、手动调用 System.gc()

2. 手动触发与注意事项

  • System.gc():建议 JVM 触发 Full GC(不保证立即执行)。

  • 风险:频繁 Full GC 会导致应用停顿(Stop-The-World),需谨慎使用。


四、垃圾回收算法与实现

1. 基础算法对比

算法步骤优点缺点适用场景
标记-清除标记存活对象 → 清除未标记对象简单内存碎片化老年代(CMS)
复制算法存活对象复制到新区域 → 清空原区域无碎片,高效内存利用率 50%新生代(Survivor)
标记-整理标记存活对象 → 整理到内存一端无碎片化整理耗时老年代(Serial Old)

2. 分代收集理论

  • 弱分代假说:绝大多数对象朝生夕灭(新生代)。

  • 强分代假说:熬过多次 GC 的对象难以消亡(老年代)。

  • 分代设计

    • 新生代:使用复制算法(Eden + Survivor)。

    • 老年代:使用标记-清除或标记-整理算法。

3. 新生代回收:Apple式复制算法

  • 内存划分

    • Eden : Survivor1 : Survivor2 = 8:1:1(默认)。

  • 回收流程

    1. 新对象分配至 Eden 区。

    2. Eden 满时触发 Minor GC,存活对象复制到 Survivor1。

    3. 下次 Minor GC 时,Eden 和 Survivor1 存活对象复制到 Survivor2,并清空原区域。

    4. 对象年龄达到阈值(默认 15)后晋升老年代。


五、主流垃圾收集器详解

1. CMS 收集器(低停顿优先)

  • 目标:最小化应用停顿时间。

  • 算法:标记-清除。

  • 工作流程

    1. 初始标记(STW):标记 GC Roots 直接关联对象。

    2. 并发标记:遍历对象图(与用户线程并发)。

    3. 重新标记(STW):修正并发标记期间变动的引用。

    4. 并发清除:清理垃圾(与用户线程并发)。

  • 缺点

    • 内存碎片化(需定期 Full GC 整理)。

    • 并发阶段占用 CPU 资源。

2. G1 收集器(平衡吞吐与延迟)

  • 目标:可预测的停顿时间(如 200ms 内)。

  • 内存布局:将堆划分为多个 Region(默认 2048 个)。

  • 工作流程

    1. 初始标记(STW):标记 GC Roots 直接关联对象。

    2. 并发标记:遍历对象图(与用户线程并发)。

    3. 最终标记(STW):处理剩余引用变更。

    4. 筛选回收(STW):选择性价比高的 Region 回收。

  • 优势

    • 支持大内存(TB 级)。

    • 通过 Region 划分减少碎片化。

3. 收集器对比

收集器算法区域特点适用场景
CMS标记-清除老年代低停顿,但碎片化严重响应敏感型应用
G1标记-整理全堆可预测停顿,兼顾吞吐与延迟大内存、低延迟应用

六、调优建议与工具推荐

1. 参数配置示例

# 使用 G1 收集器,堆内存 4G,目标停顿 200ms
java -Xmx4G -Xms4G -XX:+UseG1GC -XX:MaxGCPauseMillis=200 -jar app.jar

# 启用 CMS 收集器
-XX:+UseConcMarkSweepGC -XX:+CMSIncrementalMode

2. 常见问题排查

  • 频繁 Full GC

    • 检查内存泄漏(如静态集合未清理)。

    • 调整新生代与老年代比例(-XX:NewRatio)。

  • 长时间 STW

    • 切换低延迟收集器(如 G1/ZGC)。

    • 减少堆内存大小(权衡吞吐与停顿)。

3. 工具推荐

  • 监控工具:VisualVM、JConsole、Prometheus + Grafana。

  • 日志分析:GCeasy、GCViewer。

  • 诊断工具:Arthas、MAT(Memory Analyzer Tool)。


七、总结

  • 生死判定:可达性分析是核心,finalize() 是最后的逃生机会。

  • 引用分级:软、弱引用优化内存敏感场景。

  • 算法选择:分代理论平衡效率与资源利用率。

  • 收集器选型:CMS 适合低延迟,G1 适合大内存与可预测停顿。

核心原则:结合业务需求与监控数据动态调优,避免盲目配置。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/983698.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

【Python编程】高性能Python Web服务部署架构解析

一、FastAPI 与 Uvicorn/Gunicorn 的协同 1. 开发环境&#xff1a;Uvicorn 直接驱动 作用&#xff1a;Uvicorn 作为 ASGI 服务器&#xff0c;原生支持 FastAPI 的异步特性&#xff0c;提供热重载&#xff08;--reload&#xff09;和高效异步请求处理。 启动命令&#xff1a; u…

Sentinel 笔记

Sentinel 笔记 1 介绍 Sentinel 是阿里开源的分布式系统流量防卫组件&#xff0c;专注于 流量控制、熔断降级、系统保护。 官网&#xff1a;https://sentinelguard.io/zh-cn/index.html wiki&#xff1a;https://github.com/alibaba/Sentinel/wiki 对比同类产品&#xff1…

JQuery 语法 $

jQuery 语法是通过选取 HTML 元素, 并对选取的元素执⾏某些操作 JQuery 选择器 jQuery 中所有选择器都以 $ 开头&#xff1a;$(). JQuery事件 事件由三部分组成: 1. 事件源: 哪个元素触发的 2. 事件类型: 是点击, 选中, 还是修改? 3. 事件处理程序: 进⼀步如何处理. …

算法每日一练 (9)

&#x1f4a2;欢迎来到张胤尘的技术站 &#x1f4a5;技术如江河&#xff0c;汇聚众志成。代码似星辰&#xff0c;照亮行征程。开源精神长&#xff0c;传承永不忘。携手共前行&#xff0c;未来更辉煌&#x1f4a5; 文章目录 算法每日一练 (9)最小路径和题目描述解题思路解题代码…

2025/3/8 第 27 场 蓝桥入门赛 题解

1. 38红包【算法赛】 签到题&#xff1a; 算倍数就行了 #include <bits/stdc.h> using namespace std; int main() {int ans0;for(int i1;i<2025;i){if(i % 3 0)ans;else if(i % 8 0)ans;else if(i % 38 0)ans;}cout<<ans<<endl;return 0; } 2. 祝福…

《白帽子讲 Web 安全》之深入同源策略(万字详解)

目录 引言 一、同源策略基础认知 &#xff08;一&#xff09;定义 &#xff08;二&#xff09;作用 &#xff08;三&#xff09;作用机制详解 二、同源策略的分类 &#xff08;一&#xff09;域名同源策略 &#xff08;二&#xff09;协议同源策略 &#xff08;三&…

基于SpringBoot的商城管理系统(源码+部署教程)

运行环境 数据库&#xff1a;MySql 编译器&#xff1a;Intellij IDEA 前端运行环境&#xff1a;node.js v12.13.0 JAVA版本&#xff1a;JDK 1.8 主要功能 基于Springboot的商城管理系统包含管理端和用户端两个部分&#xff0c;主要功能有&#xff1a; 管理端 首页商品列…

FFmpeg-chapter7和chapter8-使用 FFmpeg 解码视频(原理篇和实站篇)

解码流程如下图 流程&#xff1a;首先&#xff0c;通过 avcodec_alloc_context3(nullptr) 分配一个 AVCodecContext 结构体&#xff0c;然后使用 avcodec_parameters_to_context 将参数复制到上下文中&#xff0c;接着通过 avcodec_find_decoder 查找指定的解码器&#xff0c;并…

【银河麒麟高级服务器操作系统实例】虚拟机桥接网络问题分析及处理

更多银河麒麟操作系统产品及技术讨论&#xff0c;欢迎加入银河麒麟操作系统官方论坛 https://forum.kylinos.cn 了解更多银河麒麟操作系统全新产品&#xff0c;请点击访问 麒麟软件产品专区&#xff1a;https://product.kylinos.cn 开发者专区&#xff1a;https://developer…

10 【HarmonyOS NEXT】 仿uv-ui组件开发之Avatar头像组件开发教程(一)

温馨提示&#xff1a;本篇博客的详细代码已发布到 git : https://gitcode.com/nutpi/HarmonyosNext 可以下载运行哦&#xff01; 目录 第一篇&#xff1a;Avatar 组件基础概念与设计1. 组件概述2. 接口设计2.1 形状类型定义2.2 尺寸类型定义2.3 组件属性接口 3. 设计原则4. 使用…

C++20 DR11:数组 `new` 可以推导出数组大小

文章目录 背景与动机C20 的改进示例代码编译器支持总结 在 C20 中&#xff0c;DR11 提案&#xff08;P1009R2&#xff09;引入了一项重要的语言特性改进&#xff1a;数组 new 表达式可以自动推导数组大小。这一改进极大地简化了动态数组的创建过程&#xff0c;使代码更加简洁易…

STM32-I2C通信外设

目录 一&#xff1a;I2C外设简介 二&#xff1a;I2C外设数据收发 三&#xff1a;I2C的复用端口 四&#xff1a;主机发送和接收 五&#xff1a;硬件I2C读写MPU6050 相关函数&#xff1a; 1.I2C_ GenerateSTART 2.I2C_ GenerateSTOP 3.I2C_ AcknowledgeConfig 4.I2C…

OpenManus:开源版Manus的快速安装及使用「喂饭教程」

OpenManus&#xff1a;开源版Manus的快速安装及使用「喂饭教程」 OpenManus是什么&#xff1f;OpenManus的核心理念1. 安装2. 配置2.1 线上模型2.2 本地模型 3. 运行项目常见问题&#xff1a;如何设置项目执行的Steps&#xff1f; OpenManus是什么&#xff1f; OpenManus是由 …

专业工具,提供多种磁盘分区方案

随着时间的推移&#xff0c;电脑的磁盘空间往往会越来越紧张&#xff0c;许多人都经历过磁盘空间不足的困扰。虽然通过清理垃圾文件可以获得一定的改善&#xff0c;但随着文件和软件的增多&#xff0c;磁盘空间仍然可能显得捉襟见肘。在这种情况下&#xff0c;将其他磁盘的闲置…

【小技巧】百度网盘清除重复文件详细步骤

百度网盘内存空间清理——清除重复文件 1.点击左下角【工具】 2.选择文件管理 3.点击垃圾文件清理&#xff0c;选择扫描重复文件 4.根据需要进行重复文件清理或进行垃圾视频扫描、空文件夹扫描等清理操作 5.一键清理需要svip会员&#xff0c;但是我们可以根据重复文件检查结…

用数据唤醒深度好眠,时序数据库 TDengine 助力安提思脑科学研究

在智能医疗与脑科学快速发展的今天&#xff0c;高效的数据处理能力已成为突破创新的关键。安提思专注于睡眠监测与神经调控&#xff0c;基于人工智能和边缘计算&#xff0c;实现从生理体征监测、智能干预到效果评估的闭环。面对海量生理数据的存储与实时计算需求&#xff0c;安…

运行OpenManus项目(使用Conda)

部署本项目需要具备一定的基础&#xff1a;Linux基础、需要安装好Anaconda/Miniforge&#xff08;Python可以不装好&#xff0c;直接新建虚拟环境的时候装好即可&#xff09;&#xff0c;如果不装Anaconda或者Miniforge&#xff0c;只装过Python&#xff0c;需要确保Python是3.…

upload-labs文件上传

第一关 上传一个1.jpg的文件&#xff0c;在里面写好一句webshell 保留一个数据包&#xff0c;将其中截获的1.jpg改为1.php后重新发送 可以看到&#xff0c;已经成功上传 第二关 写一个webshell如图&#xff0c;为2.php 第二关在过滤tpye的属性&#xff0c;在上传2.php后使用b…

【微知】Centos如何迁移到Anolis系统的失败记录?(yum -y install centos2anolis、centos2anolis.py)

背景 本文记录如何从centos 8迁移到anolis系统。 详细步骤 下载迁移repo wget https://mirrors.openanolis.cn/anolis/migration/anolis-migration.repo -O /etc/yum.repos.d/anolis-migration.repo下载centos2anolis工具包 yum -y install centos2anolis安装额外工具包 …

Docker 安装 Nacos 2.1.1(单机版)

一、拉取镜像 docker pull nacos/nacos-server:v2.1.1 二、新建数据库 官网上下载 对应版本的 nacos zip 包&#xff0c;在 nacos\conf 目录下有 mysql脚本&#xff1a; 新建一个数据库 nacos_config&#xff0c;在数据库中依次执行 nacos-mysql.sql、1.4.0-ipv6_support-up…