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;@Overrideprotected 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 GC | Eden 区空间不足 |
| 老年代 | 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(默认)。
-
-
回收流程:
-
新对象分配至 Eden 区。
-
Eden 满时触发 Minor GC,存活对象复制到 Survivor1。
-
下次 Minor GC 时,Eden 和 Survivor1 存活对象复制到 Survivor2,并清空原区域。
-
对象年龄达到阈值(默认 15)后晋升老年代。
-
五、主流垃圾收集器详解
1. CMS 收集器(低停顿优先)
-
目标:最小化应用停顿时间。
-
算法:标记-清除。
-
工作流程:
-
初始标记(STW):标记 GC Roots 直接关联对象。
-
并发标记:遍历对象图(与用户线程并发)。
-
重新标记(STW):修正并发标记期间变动的引用。
-
并发清除:清理垃圾(与用户线程并发)。
-
-
缺点:
-
内存碎片化(需定期 Full GC 整理)。
-
并发阶段占用 CPU 资源。
-
2. G1 收集器(平衡吞吐与延迟)
-
目标:可预测的停顿时间(如 200ms 内)。
-
内存布局:将堆划分为多个 Region(默认 2048 个)。
-
工作流程:
-
初始标记(STW):标记 GC Roots 直接关联对象。
-
并发标记:遍历对象图(与用户线程并发)。
-
最终标记(STW):处理剩余引用变更。
-
筛选回收(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 适合大内存与可预测停顿。
核心原则:结合业务需求与监控数据动态调优,避免盲目配置。
相关文章:
Java虚拟机之垃圾收集(一)
目录 一、如何判定对象“生死”? 1. 引用计数算法(理论参考) 2. 可达性分析算法(JVM 实际使用) 3. 对象的“缓刑”机制 二、引用类型与回收策略 三、何时触发垃圾回收? 1. 分代回收策略 2. 手动触发…...
【mysql】mysql数据库,权限授予,账号设置
创建一个可用的数据库,名字为gitea,支持中文utf8mb4 CREATE DATABASE gitea CHARACTER SET utf8mb4 COLLATE utf8mb4_unicode_ci; # 设置东八区时区 SET GLOBAL time_zone 08:00; FLUSH PRIVILEGES; # 查询当前时间 SELECT NOW();给【普通用户】授予查…...
Html5学习教程,从入门到精通, HTML5超链接应用的详细语法知识点和案例代码(18)
HTML5超链接应用的详细语法知识点和案例代码 超链接(Hyperlink),也称为跃点链接,是互联网和文档编辑中的一种重要概念。 超链接的定义 超链接是指从一个网页指向一个目标的连接关系,这个目标可以是另一个网页&#…...
⭐LeetCode(数学分类) 48. 旋转图像——优美的数学法转圈(原地修改)⭐
⭐LeetCode(数学分类) 48. 旋转图像——优美的数学法转圈(原地修改)⭐ 示例 1: 输入:root [5,3,6,2,4,null,8,1,null,null,null,7,9] 输出:[1,null,2,null,3,null,4,null,5,null,6,null,7,null,8,null,9] 示例 2: 输入࿱…...
深度学习PyTorch之13种模型精度评估公式及调用方法
深度学习pytorch之22种损失函数数学公式和代码定义 深度学习pytorch之19种优化算法(optimizer)解析 深度学习pytorch之4种归一化方法(Normalization)原理公式解析和参数使用 深度学习pytorch之简单方法自定义9类卷积即插即用 实时…...
tomcat单机多实例部署
一、部署方法 多实例可以运行多个不同的应用,也可以运行相同的应用,类似于虚拟主机,但是他可以做负载均衡。 方式一: 把tomcat的主目录挨个复制,然后把每台主机的端口给改掉就行了。 优点是最简单最直接,…...
Java开发者如何接入并使用DeepSeek
目录 一、准备工作 二、添加DeepSeek SDK依赖 三、初始化DeepSeek客户端 四、数据上传与查询 五、数据处理与分析 六、实际应用案例 七、总结 【博主推荐】:最近发现了一个超棒的人工智能学习网站,内容通俗易懂,风格风趣幽默ÿ…...
win10电脑鼠标速度突然变的很慢?
电脑鼠标突然变很慢,杀毒检测后没问题,鼠标设置也没变,最后发现可能是误触鼠标的“DPI”调节键。 DPI调节键在鼠标滚轮下方,再次点击即可恢复正常鼠标速度。 如果有和-的按键,速度变快,-速度变慢。 图源&…...
第四次CCF-CSP认证(含C++源码)
第四次CCF-CSP认证 第一道(easy)思路及AC代码 第二道(easy)思路及AC代码遇到的问题 第三道(mid)思路及AC代码 第一道(easy) 题目链接 思路及AC代码 这题就是将这个矩阵旋转之后输出…...
Netty基础—1.网络编程基础一
大纲 1.什么是OSI开放系统互连 2.OSI七层模型各层的作用 3.TCP/IP协议的简介 4.TCP和UDP的简介 5.TCP连接的三次握手 6.TCP连接的四次挥手 7.TCP/IP中的数据包 8.TCP通过确认应答与序列号提高可靠性 9.HTTP请求的传输过程 10.HTTP协议报文结构 11.Socket、短连接、长…...
【理想解法学习笔记】
目录 理想解法原理简介算法步骤属性值规范化方法代码示例 理想解法 原理简介 TOPSIS(Technique for Order Preference by Simi larity to IdealSolution)法是一种逼近理想解的排序方法。其基本的处理思路是:首先建立初始化决策矩阵,而后基于规范化后的初…...
98.在 Vue3 中使用 OpenLayers 根据 Resolution 的不同显示不同的地图
在 Vue3 中使用 OpenLayers 根据 Resolution 的不同显示不同的地图 前言 在 Web GIS(地理信息系统)应用开发中,地图的 Resolution(分辨率)是一个重要的概念。不同的 Resolution 适用于不同的地图层级,有时…...
Docker 部署 Vaultwarden
一、前言 1. 官网 1.1 Vaultwarden https://github.com/dani-garcia/vaultwarden https://github.com/wcjxixi/Vaultwarden-Wiki-Chn https://hub.docker.com/r/vaultwarden/server https://rs.ppgg.in/ # Vaultwarden Wiki 中文版 https://geekdaxue.co/read/Vaultward…...
Smart contract -- 自毁合约
在区块链开发中,Solidity 语言提供了强大的功能,其中自毁合约是一个独特且重要的特性。今天,就让我们深入探讨一下 Solidity 中的自毁合约,以及如何使用 selfdestruct 函数。 注意:使用继承时请确保代码的正确性&#…...
unity学习64,第3个小游戏:一个2D跑酷游戏
目录 学习参考 素材资源导入 1 创建项目 1.1 创建1个2D项目 1.2 导入素材 2 背景图bg 2.0 bg素材 2.1 创建背景 2.2 修改素材,且修改摄像机等 2.2.1 修改导入的原始prefab素材 2.2.2 对应调整摄像机 2.2.3 弄好背景 2.3 背景相关脚本实现 2.3.1 错误…...
Python Flask 在网页应用程序中处理错误和异常
Python Flask 在网页应用程序中处理错误和异常 Python Flask 在网页应用程序中处理错误和异常 Python Flask 在网页应用程序中处理错误和异常 在我们所有的代码示例中,我们没有注意如何处理用户在浏览器中输入错误的URL或向我们的应用程序发送错误的参数集的情况。…...
模板方法模式的C++实现示例
核心思想 模板方法设计模式是一种行为设计模式,它定义了一个算法的框架,并将某些步骤的具体实现延迟到子类中。通过这种方式,模板方法模式允许子类在不改变算法结构的情况下重新定义算法的某些步骤。 模板方法模式的核心在于: …...
水下机器人推进器PID参数整定与MATLAB仿真
水下机器人推进器PID参数整定与MATLAB仿真 1. PID控制原理 目标:通过调节比例(P)、积分(I)、微分(D)参数,使推进器输出力快速稳定跟踪期望值。传递函数(示例):推进器动力学模型可简化为: [ G(s) = \frac{K}{\tau s + 1} \cdot e^{-Ts} ] 其中:K为增益,τ为时间常…...
在本地部署DeepSeek等大模型时,需警惕的潜在安全风险
在本地部署DeepSeek等大模型时,尽管数据存储在本地环境(而非云端),但仍需警惕以下潜在安全风险: 1. 模型与数据存储风险 未加密的存储介质:若训练数据、模型权重或日志以明文形式存储,可能被物…...
智能焊机监测系统:打造工业安全的数字化盾牌
在现代工业生产中,焊机作为核心设备之一,其稳定性和安全性直接关系到生产效率和产品质量。德州迪格特科技有限公司推出的智能焊机监测系统,通过先进的技术手段,为工业生产构筑了一道坚固的安全防线。 智能监测,保障焊…...
【redis】string类型相关操作:SET、GET、MSET、MGET、SETNX、SETEX、PSETEX
文章目录 二进制存储编码转换SET 和 GETSETGET MSET 和 MGETSETNX、SETEX 和 PSETEX Redis 所有的 key 都是字符串,value 的类型是存在差异的 二进制存储 Redis 中的字符串,直接就是按照二进制数据的方式存储的 不仅仅可以存储文本数据,还可…...
GaussDB安全配置指南:从认证到防御的全方面防护
一、引言 随着企业数据规模的扩大和云端化进程加速,数据库安全性成为运维的核心挑战之一。GaussDB作为一款高性能分布式数据库,提供了丰富的安全功能。本文将从 认证机制、权限控制、数据加密、审计日志 等维度,系统性地讲解如何加固 Ga…...
总结学习课程
1. 数据加载与预处理 PyTorch工具加载和预处理数据(如MNIST数据集)。 2. 定义模型 - 使用nn.Module构建神经网络,定义各层和前向传播。 3. 损失函数与优化器: 选择损失函数(如交叉熵损失)和优化…...
Ubuntu20.04搭建gerrit code review
一、环境准备 1. 安装 Java 环境 Gerrit 依赖 Java 运行环境(推荐 JDK 8): sudo apt install openjdk-11-jdk 验证安装: java -version 2. 安装 Git sudo apt install git 3. 可选依赖 数据库:Gerrit …...
MacOS安装FFmpeg和FFprobe
按照网上很多教程安装,结果都失败了,后来才发现是路径问题,其实安装过程很简单(无奈) 第一步: 在官网下载 打开页面后,可以看到FFmpeg、FFprobe、FFplay和FFserver的下载图标 第二步࿱…...
Redis7系列:设置开机自启
前面的文章讲了Redis和Redis Stack的安装,随着服务器的重启,导致Redis 客户端无法连接。原来的是Redis没有配置开机自启。此文记录一下如何配置开机自启。 1、修改配置文件 前面的Redis和Redis Stack的安装的文章中已经讲了redis.config的配置…...
SpringAI介绍及本地模型使用方法
博客原文地址 前言 Spring在Java语言中一直稳居高位,与AI的洪流碰撞后也产生了一些有趣的”化学反应“,当然你要非要说碰撞属于物理反应也可以, 在经历了一系列复杂的反应方程后,Spring家族的新成员——SpringAI,就…...
Zookeeper实践指南
Zookeeper实践指南 1. 什么是 Zookeeper? Zookeeper 是 Apache 旗下的一个开源分布式协调框架,主要用于解决分布式系统中的一致性问题,提供高效可靠的分布式数据管理能力。 1.1 Zookeeper 的核心特性 顺序一致性:客户端的更新…...
Unity 基础知识总结(持续更新中...)
引擎基础 Unity有哪几个主要窗口? Scene窗口 用于场景搭建和UI界面拼接 Game窗口 游戏运行预览 Hierarchy窗口 查看和调整场景对象层级结构 Project窗口 游戏工程资源 Inspector创建 属性查看器,属性设置、脚本组件挂载 Unity提供了几种光源…...
IDEA接入阿里云百炼中免费的通义千问[2025版]
安装deepseek 上一篇文章IDEA安装deepseek最新教程2025中说明了怎么用idea安装codeGPT插件,并接入DeepSeek,无奈接入的官方api已经不能使用了,所以我们尝试从其他地方接入 阿里云百炼https://bailian.console.aliyun.com/ 阿里云百炼是阿…...
