当前位置: 首页 > news >正文

【基础篇】十二、引用计数法 可达性分析算法

文章目录

  • 1、Garbage Collection
  • 2、方法区的回收
  • 3、堆对象回收
  • 4、引用计数法
  • 5、可达性分析算法
  • 6、查看GC Root对象

在这里插入图片描述

1、Garbage Collection

C/C++,无自动回收机制,对象不用时需要手动释放,否则积累导致内存泄漏:

在这里插入图片描述
Java、C#、Python、Go等支持对不再使用的对象进行回收:

在这里插入图片描述

Java引入GC机制,主要负责对上的内存进行回收,简化了对象的释放,但同时也丧失了回收的及时性,因为回收操作不再又开发者做了。

在这里插入图片描述

线程不共享的部分,随着线程的创建而创建,随着线程的死亡而销毁,不会发生内存泄漏。且栈里的每个栈帧,在方法执行完后直接弹栈并释放内存,都不用等到线程死亡。

在这里插入图片描述
添加JVM参数:

-verbose:gc

可查看回收日志

2、方法区的回收

即回收不再使用的类,对应类生命周期里的卸载阶段。类A被卸载,需同时满足

  • 类的所有实例对象全被回收,堆中再无它的对象或子类的对象
  • 加载类A的类加载器被回收
  • 类A的java.lang.Class对象没有任何地方在引用
URLClassLoader loader = new URLClassLoader(new URL[] {new URL("file:D:\\lib\\")});
Class<?> clazz = loader.loadClass("com.plat.domain.A");
Object a = clazz.newInstance();
a = null;
clazz = null;
loader = null;

Demo代码如下,循环体中创建的变量,一轮结束后自动没用,不用重复 o = null

public class Demo2 {public static void main(String[] args) {try {ArrayList<Class<?>> classes = new ArrayList<>();ArrayList<ClassLoader> loaders = new ArrayList<>();ArrayList<Object> objs = new ArrayList<>();while (true) {//URLClassLoader classLoader = new URLClassLoader(new URL[]{new URL("file:D:\\tmp\\")});BreakClassLoader1 classLoader = new BreakClassLoader1();  //自定义类加载器classLoader.setBasePath("D:\\springboot\\jvm\\target\\classes\\");Class<?> clazz = classLoader.loadClass("com.plat.app.Demo1");Object o = clazz.newInstance();//循环体中创建的变量,一轮结束后自动没用,不用重复 o = null 、clazz = null 、classLoader = null/*打开这里的任意一句,就有一个对应的对象被引用,类卸载就会失败objs.add(o);loaders.add(classLoader);classes.add(clazz);*///手动触发一次垃圾回收,但不是立即//是对JVM的请求,具体是否回收JVM自行决断System.gc();}} catch (Exception e) {e.printStackTrace();}}}public class Demo1  {static{System.out.println("类A被加载");}
}

添加JVM参数:

//运行过程中打印类的加载信息
-XX:+TraceClassLoading   //类被卸载的时候打印一句日志
-XX:+TraceClassUnloading  

运行:

在这里插入图片描述

但其实类的卸载不常见,因为自定义类的累加载器对象还在,热部署中用的多:

在这里插入图片描述

3、堆对象回收

Java对象能否被回收,得看它是否还在被引用。

在这里插入图片描述
demo = null后,再无对Demo对象的引用,可回收。

在这里插入图片描述
执行:

a1 = null
b1 = null

A、B对象被回收,因为方法中无法再通过引用访问到这两个对象了。总之,判断方式:

  • 引用计数法
  • 可达性分析

4、引用计数法

即为每个对象维护一个计数器,对象被引用就+1,置为null了就-1,JVM扫描堆内存,发现数值为0则回收

A a1 = new A();
B b1 = new B():
a1.SetB(b1);

则A的计数器为1,B的计数器为2
在这里插入图片描述
优点:

  • 简单
  • 也是C++的智能指针的实现基础

缺点:

  • +1,-1的维护损耗性能
  • 循环引用时,导致计数器至少为1,对象无法被回收,内存泄漏

在这里插入图片描述

5、可达性分析算法

普通对象A,经一个引用链可以到达GC Root对象,则A不可被回收

在这里插入图片描述

  • GC Root对象即垃圾回收的根对象
  • GC Root对象一般不会被回收,且JVM持有GC Root对象的List列表

GC Root对象包括:

  • 线程对象Thread(引用线程栈帧中的方法参数、局部变量等)

在这里插入图片描述

  • 系统类加载器加载的java.lang.Class对象(引用类中的静态变量)

在这里插入图片描述

  • 监视器对象(引用synchronized锁的对象)

在这里插入图片描述

  • native本地方法调用时使用的全局对象

6、查看GC Root对象

步骤:

  • 阿尔萨斯的heapdump指令,保存堆内存快照到本地磁盘
heapdump d:/tmp/test.hprof
  • 使用MAT工具打开堆内存快照文件

在这里插入图片描述

http://www.lryc.cn/news/276145.html

相关文章:

  • C语言算法(二分查找、文件读写)
  • 流媒体学习之路(WebRTC)——Pacer与GCC(5)
  • 2023版本QT学习记录 -11- 多线程的使用(QT的方式)
  • iOS苹果和Android安卓测试APP应用程序的差异
  • 每日算法打卡:数的三次方根 day 7
  • 人机交互主板定制_基于MT8735安卓核心板的自助查询机方案
  • 全志F1C100s Linux 系统编译出错:不能连接 github
  • 如何保障 MySQL 和 Redis 的数据一致性?
  • Leetcode 2999. Count the Number of Powerful Integers
  • 【Reading Notes】(2)
  • 【设计模式之美】SOLID 原则之一:怎么才算是单一原则、如何取舍单一原则
  • # [NOIP2015 普及组] 扫雷游戏#洛谷
  • Unity中Shader的_Time精度问题
  • 听GPT 讲Rust源代码--compiler(15)
  • 关键字联合体union的定义和使用
  • 基于GA-PSO遗传粒子群混合优化算法的VRPTW问题求解matlab仿真
  • 【leetcode100-033】【链表】排序链表
  • [Kubernetes]5. k8s集群StatefulSet详解,以及数据持久化(SC PV PVC)
  • 数据库系统-甘晴void学习笔记
  • Azure Machine Learning - 人脸识别任务概述与技术实战
  • 强化学习的数学原理学习笔记 - 蒙特卡洛方法(Monte Carlo)
  • DDIA 第十一章:流处理
  • webpack知识点总结(高级应用篇)
  • 均匀与准均匀 B样条算法
  • 2023年12 月电子学会Python等级考试试卷(一级)答案解析
  • 启发式算法解决TSP、0/1背包和电路板问题
  • 阿里云新用户的定义与权益
  • go语言多线程操作
  • GreatSQL社区2023全年技术文章总结
  • 【论文阅读笔记】Stable View Synthesis 和 Enhanced Stable View Synthesis