JVM-垃圾回收器与内存分配策略详解

1.如何判断对象已死

1.1 对象引用的4种类型(强软弱虚)

1.1.1 强引用

特点:最常见的引用类型,只要强引用存在,对象绝不会被回收

Object strongObj = new Object();  // 强引用

注意:集合类型,如果对象不再使用,应当及时清除,避免内存泄漏

1.1.2 软引用

特点:内存不足时才会被回收,适合实现缓存

SoftReference<byte[]> softRef = new SoftReference<>(new byte[1024*1024*10]); // 10MB

 适用场景:

  1. 缓存(EhCache中使用软引用,一旦触发内存不足时,就会由JVM自动清理数据,释放内存)

缓存本身的含义:可删除

1.1.3 弱引用

特点:下次GC时必定被回收,适合维护非必要数据

 // 弱引用基本示例
WeakReference<String> weakRef = new WeakReference<>(new String("Weak Object"));     

 适用场景:

        临时的数据

1.1.4 虚引用

1.1.5 其他

1.1.5.1 ReferenceQueue

referenceQueue(引用队列)是Java引用类型体系中的关键组件,与SoftReference、WeakReference和PhantomReference配合使用,提供了对象生命周期监控的机制。

  1.  通知机制:当被引用的对象达到相应回收状态时,引用对象本身会进入队列
  2.   非阻塞设计:提供poll()和remove()两种获取方式
  3.  GC协作:由垃圾收集器在适当时候自动入队

 工作流程

 创建引用对象 → 关联ReferenceQueue → 对象被回收 → 引用对象入队 → 客户端处理
引用类型入队时机典型用途
虚引用对象被回收且finalize完成后精准资源清理
弱引用对象被GC回收时缓存/临时映射
软引用内存不足被回收时内存敏感缓存
强引用从不入队常规对象引用

 

1.1.6 使用案例

软引用缓存,清除无效的key


/*** 弱引用缓存*/public class WeakCache<K, V> {private final ConcurrentHashMap<K, WeakReference<V>> cache = new ConcurrentHashMap<>();private final ReferenceQueue<V> queue = new ReferenceQueue<>();public void put(K key, V value) {cleanUp();cache.put(key, new WeakReference<>(value, queue));}// 清除缓存中已释放的引用(原因:value是弱引用、但是key是强引用,value被释放了,key没有作用,应当也被释放)private void cleanUp() {Reference<? extends V> ref;while ((ref = queue.poll()) != null) {Reference<? extends V> finalRef = ref;cache.values().removeIf(weakRef -> weakRef == finalRef);}}
}

1.2 判断对象已死

1.2.1 方案1:引用计数算法

核心思想
  • 每个对象维护一个引用计数器,记录有多少引用指向它
计数规则
  • 新引用指向对象时,计数器+1
  • 引用失效时,计数器-1
  • 计数器=0时立即回收对象

1.2.2 方案2:可达性分析算法

核心思想
  • 通过GC Roots作为起点,遍历对象引用链
判定规则
  • 从GC Roots不可达的对象判定为垃圾
  • 通常在特定时间点(如堆空间不足时)执行
GC Roots有哪些

GC Roots是垃圾回收的起点,所有从这些根对象直接或间接可达的对象都被视为存活对象。以下是Java中常见的GC Roots类型:

  • 虚拟机栈(栈帧中的本地变量表)中的引用对象
public void method() {Object localObj = new Object();  // localObj是GC Root// 方法执行期间,localObj引用的对象不会被回收   
}当前所有正在执行的方法中的局部变量和参数包括Java方法、native方法的栈帧中的引用
  • ​​​方法区中类静态属性引用的对象
class MyClass {static Object staticObj = new Object();  // staticObj是GC Root
}特点:类的静态变量引用的对象直到ClassLoader卸载前都会保持强引用
  • 方法区中常量引用的对象
class MyClass {static final String CONSTANT = "常量";  // CONSTANT是GC Root
}特点:包括字符串常量池中的引用编译期常量(constantValue属性)不算真正的GC Root

  • 本地方法栈中JNI(Java Native Interface)引用的对象
public native void nativeMethod(Object obj);  // native代码中的obj引用Java调用native方法时传入的参数对象
全局JNI引用(NewGlobalRef创建)也是GC Root
  • 被同步锁(synchronized)持有的对象
  • Java虚拟机内部的引用

包括:

  • 基本类型对应的Class对象

  • 常驻异常对象(NullPointerException等)

  • 系统类加载器

  • 类加载器管理的Class对象

1.2.3 工作原理对比

特征引用计数可达性分析
触发时机实时(引用变更时)周期性/按需
执行速度分散的微小停顿集中式停顿(STW问题)
内存回收立即回收延迟回收
实现复杂度简单复杂
对象判定计数器=0GC Roots不可达

 

1.2.4 优缺点分析

1.2.4.1 引用计数的优缺点

优点

  1. 立即回收内存,减少内存占用
  2. 垃圾回收开销平均分配到程序运行中
  3. 不需要STW(Stop-The-World)暂停

缺点:

  • 计数器维护带来额外内存开销
  • 频繁更新计数器影响性能
  • 循环引用问题(主要缺陷)
// 引用计数无法处理的场景   
class A { B b; }   
class B { A a; }void createCycle() {A a = new A();  // A计数=1B b = new B();  // B计数=1a.b = b;        // B计数=2b.a = a;        // A计数=2// 即使外部引用消失...a = null;       // A计数=1b = null;       // B计数=1// 对象仍无法回收!   
}// 可达性分析可以正确处理
1.2.4.2 可达性分析的优缺点

优点

  1. 能正确处理循环引用
  2. 回收效率高(集中处理)
  3. 现代JVM采用的成熟方案

缺点

  1. 需要STW暂停应用线程
  2. 实现复杂度高(需要准确识别GC Roots)
  3. 内存回收不及时

java采用可达性分析算法

2.垃圾收集算法

分代收集理论

标记-清除算法

标记-复制算法

标记-整理算法

算法细节(根节点枚举、安全点、安全区域、记忆集与卡表、写屏障、并发的可达性分析)

3.常见的垃圾收集器

3.1 serial

3.2 parNew

3.3 Parallel Scavenge

3.4 serial old

3.5 Parallel Old

3.6 CMS

3.7 G1

3.8 ZGC

4. 内存分配策略

对象优先分配到Eden区

大对象直接进入老年代

长期存活的对象将进入老年代

动态对象年龄判断

空间分配担保

垃圾回收的优化案例

fullGc问题

对于绝对不可接受Full GC的系统:

  1. 商用解决方案:Azul Zing(C4 GC)、IBM Balanced GC
  2. 内存架构改造
    • 堆外内存存储(HBase/OFF-Heap)
    • 分布式缓存集群
  3. 服务拆分:将内存敏感组件独立部署

最佳实践:某证券交易所系统通过G1+堆外缓存方案,实现连续18个月零Full GC,老年代内存通过并发标记和定期混合回收保持85%以上可用率。

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

新手向:简易Flask/Django个人博客

从零开始搭建个人博客:Flask与Django双版本指南 本文将详细讲解如何使用两种主流Python框架构建功能完整的个人博客系统。我们将从零开始,分别使用轻量级的Flask框架和功能全面的Django框架实现以下核心功能: 用户认证系统: 用户注册/登录/注销功能 密码加密存储 会话管理…

使用 Trea cn 设计 爬虫程序 so esay

使用 Trea cn 设计 爬虫程序 so esay 在现代数据驱动的时代&#xff0c;网络爬虫已成为数据采集的重要工具。传统的爬虫开发往往需要处理复杂的HTTP请求、HTML解析、URL处理等技术细节。而借助 Trea CN 这样的AI辅助开发工具&#xff0c;我们可以更高效地构建功能完善的爬虫程…

MySQL Redo Log

MySQL Redo Log MySQL主从复制&#xff1a;https://blog.csdn.net/a18792721831/article/details/146117935 MySQL Binlog&#xff1a;https://blog.csdn.net/a18792721831/article/details/146606305 MySQL General Log&#xff1a;https://blog.csdn.net/a18792721831/artic…

项目实战1:Rsync + Sersync 实现文件实时同步

项目实战&#xff1a;Rsync Sersync 实现文件实时同步 客户端中数据发生变化&#xff0c;同步到server端&#xff08;备份服务器&#xff09;。 Rsync&#xff1a;负责数据同步&#xff0c;部署在server端。 Sersync&#xff1a;负责监控数据目录变化&#xff0c;并调用rsync进…

Spring Boot 全 YAML 配置 Liquibase 教程

一、项目初始化配置 1.1 创建 Spring Boot 项目 通过 Spring Initializr 生成基础项目&#xff0c;配置如下&#xff1a; ​​Project​​: Maven​​Language​​: Java​​Spring Boot​​: 3.5.3&#xff08;最新稳定版&#xff09;​​Project Metadata​​: Group: com…

STM32-驱动OLED显示屏使用SPI(软件模拟时序)实现

本章概述思维导图&#xff1a;SPI通信协议SPI通信协议介绍SPI通讯&#xff1a;高速的、串行通讯、全双工、同步、总线协议&#xff1b;&#xff08;通过片选信号选中设备&#xff09;&#xff1b;注&#xff1a;SPI通讯通过片选信号选中设备&#xff0c;串口通讯通过端口号选中…

Easy系列PLC相对运动指令实现定长输送(ST源代码)

汇川Easy系列PLC总线伺服转矩控制功能块 Easy系列PLC总线伺服转矩控制功能块(详细PDO配置+完整ST源代码)_pdo中添加目标力矩然后映射到轴中-CSDN博客文章浏览阅读215次。Easy系列PLC如何实现轮廓速度模式PV速度模式Easy系列PLC如何实现轮廓速度模式PV速度控制_汇川easy plc轮廓…

SpringCloud学习第一季-4

目录 16.SpringCloud Alibaba Nacos服务注册和配置中心 SpringCloud Alibaba简介 1. 为什么出现 SpringCloud Alibaba 2. SpringCloud Alibaba带来了什么 2.1 能干什么 2.2 去哪里下载 2.3 怎么玩 3. 学习资料的获取 17.SpringCloud Alibaba Nacos服务注册和配置中心 …

嵌入式开发学习———Linux环境下数据结构学习(五)

折半查找&#xff08;二分查找&#xff09;适用于已排序的数组&#xff0c;通过不断缩小查找范围定位目标值。int binarySearch(int arr[], int size, int target) {int left 0, right size - 1;while (left < right) {int mid left (right - left) / 2;if (arr[mid] t…

(一)React +Ts(vite创建项目/useState/Props/Interface)

文章目录 项目地址 一、React基础 1.1 vite创建 1. 创建项目 2. 安装项目所需环境 1.2 jsx 1. 三元表达式 1.3 基础 1. 创建第一个组件 2. 安装boostrap 3. 插件常用命令 4. map 二、组件 2.1 useState 1. useState 2. 使用 3.更新对象 4. 更新数组(增,删,改) 5. 使用immer…

网关和BFF是如何演化的

BFF&#xff08;Backend For Frontend&#xff09;:对返回的数据结构进行聚合、裁剪、透传等适配逻辑。 适用于API网关之后的数据聚合、裁剪与透传简化客户端逻辑&#xff0c;减少网络开销敏感数据过滤 BFF逻辑层 架构没有最好&#xff0c;要看是否满足当前的业务场景。 业务的…

SQL中的WITH语句(公共表表达式CTE)解释

SQL中的WITH语句&#xff08;公共表表达式CTE&#xff09; WITH语句&#xff0c;也称为公共表表达式&#xff08;Common Table Expression&#xff0c;CTE&#xff09;&#xff0c;是SQL中一种强大的功能&#xff0c;它允许你创建临时结果集&#xff0c;这些结果集可以在后续的…

服务器地域选择指南:深度分析北京/上海/广州节点对网站速度的影响

更多云服务器知识&#xff0c;尽在hostol.com你准备开一个覆盖全国的线上零食店&#xff0c;现在万事俱备&#xff0c;只差一个核心问题没解决&#xff1a;你唯一的那个总仓库&#xff0c;应该建在哪里&#xff1f;是建在哈尔滨&#xff0c;让南方的朋友下单后&#xff0c;一包…

桶排序-Java实现

桶排序是一种分配式排序算法&#xff0c;将元素分到有限数量的桶里&#xff0c;每个桶再单独排序&#xff08;比如用插入排序&#xff09;&#xff0c;最后依次把各个桶中的元素取出来即完成排序。 时间复杂度&#xff1a;最佳 O(n) | 平均 O(n n/k k) | 最差 O(n) 空间复杂…

oracle知识

这里写自定义目录标题Oracle常用的数据类型&#xff1a;Oracle实操&#xff1a;创建数据表Oracle约束建表的时候设置约束&#xff1a;表创建后添加添加约束&#xff1a;Oracle常用的数据类型&#xff1a; Oracle实操&#xff1a;创建数据表 Oracle约束 建表的时候设置约束&…

超级人工智能+无人机操控系统,振兴乡村经济的加速器,(申请专利应用),严禁抄袭!

无人机边缘智能系统&#xff1a;山林珍稀资源探测的完整架构与实战指南本人设计的多模态边缘AI系统已在秦岭山区完成实地验证&#xff0c;对7种高价值食用菌识别准确率达94.3%&#xff0c;定位误差小于0.8米一、前沿技术融合的商业化机遇根据Gartner 2025年技术成熟度曲线分析&…

用腾讯地图写一个逆地址解析(很详细)

首先说明以下代码适合有前端基础知识的同学。以下是css和html部分<!DOCTYPE html><html lang"zh-CN"><!-- lang是用来申明语言类型&#xff0c;这里申明为中文&#xff08;zh&#xff09;中国大陆&#xff08;CN&#xff09;补充中文繁体为zh-TW --&g…

在 Vue3+Vite+TypeScript 项目中使用 svg 文件并支持自定义样式

参考文档&#xff1a;vite-svg-loader 安装与配置 安装插件 pnpm add vite-svg-loader -D配置 // vite.config.ts import svgLoader from vite-svg-loaderexport default defineConfig({plugins: [vue(),svgLoader({defaultImport: component})] })使用 <script setup …

ShimetaPi M4-R1:国产高性能嵌入式平台的异构计算架构与OpenHarmony生态实践

在全球化芯片供应链波动及树莓派等硬件持续涨价的背景下&#xff0c;ShimetaPi M4-R1 作为全栈国产化嵌入式开发平台&#xff0c;以 高性能异构计算架构 和 开源鸿蒙原生支持 为核心突破点&#xff0c;填补了中高端边缘设备开发的国产方案空白。其基于瑞芯微 RK3568B2 的硬件设…

zookeeper分布式锁 -- 读锁和写锁实现方式

读锁和写锁读锁: 是共享锁,读锁与读锁是可以兼容的,所以同时有多个请求都可以持有写锁: 是独占锁,写锁与任何锁都互斥,所以只有一个请求持有,这个请求释放写锁其他请求才能持有一旦持有写锁,说明数据在发送变化就不能读了,自然一个请求就不能出现读锁和写锁共存的情况总结: 读锁…