-
解析(该解析大都是参考书上的P68页往后进行总结的): JVM的内存结构包含五大区域:程序计数器、虚拟机栈、本地方法栈(随线程而生,随线程而灭。这三个内存分配和回收都兼备确定性,所以不考虑回收的问题)、堆、方法区
-
程序计数器:可以理解为当前线程所执行的字节码的行号指示器,线程私有,当执行java方法的时候他记录的是正在执行的虚拟机指令的地址,当为native方法时,为空
-
虚拟机栈:java线程执行的内存,线程私有
-
本地方法栈:为本地方法服务的,线程私有
-
堆:线程共享,存放对象实例
-
方法区:线程共享,存储被虚拟机加载的类型信息、常量、静态变量…
-
运行时常量池:方法区的一部分,
首先,回收前我们必须确定那些对象可以回收、哪些不能回收,哪些暂时不能回收!这就用到判断对象是否存活的算法
-
引用计数法( Reference Counting):这个方法JAVA现在不使用, 原理:在每个对象里面都加一个计数器,当对象被引用时计数器 + 1 引用失效的时候 -1 计数器为 0 的对象就是“已死” 的 对象 缺点就是无法解决对象之间相互循环引用的问题!
-
可达性分析算法:从GC ROOT 到这个对象不可达时,则证明这个对象是不能再+使用的。可作为GC ROOT 的对象有以下几种:
-
在虚拟机栈(栈帧中的本地变量表)中引用的对象
-
方法区中类静态属性引用的变量
-
方法区中,常量引用的对象
-
本地方法栈中Native引用的对象
-
java 虚拟机内部的引用…(具体详见P70)。
算法思路:以GC ROOT 的对象作为起始点,从这些节点开始向下搜索,走过的路径称为引用链( Reference Chain ) 当一个对象到GC ROOT 时没有任何引用链就会被判”缓刑“,要想真正死亡需要经历两次标记,刚才所说的只是暂时标记了一次,第二次标记的时候系统需要判断该对象是否有必要执行一次finalize() 方法,如果对象没有覆盖,或者是已经调用了一次finalize() 方法 那就判定为不用执行,这时候直接回收了
-
关于引用:四种:强引用(不会被回收)、软引用(引用那些,有用但是非必须的对象。栈溢出异常之前会回收软引用中的对象)、弱引用(也是非必须的对象,但是它更加弱,当垃圾收集机制开始时,弱引用中的对象都会被回收)、虚引用
-
回收方法区:回收的是废弃的常量和不再被引用的类型,废弃的常量很好理解,但是不再被引用的类型需要满足一下三个条件:
-
该类的所有的实例全部都已经被回收
-
该类的类加载器也被回收
-
java.lang.Class 对象在任何地方都没有被引入
-
常用的垃圾收集算法:
2. 标记清除算法(Mark-Sweep):可以标记存活对象,也可标记死亡对象。缺点就是容易产生大量的内存碎片
-
复制算法(Copying):把内存区分成两部分,每次只是用其中的一部分,当这一部分用完之后,就会把还存活的对象复制到另一部分,然后把之前用的全部删了,这样就不容易产生内存碎片了,当然缺点也是显而易见,他的效率很低
-
标记整理算法(Mark-compact):类似Mark-Sweep ,他也会进行标记,但是标记之后他不会立马去清除可回收的对象,而是先把存活的对象都向一端进行移动,然后再清理掉边界以外的内存
-
分代收集理论:Eden 、Young Generation (Minor GC)、Old Generation(Magor GC) 。
最后
自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。
深知大多数Java工程师,想要提升技能,往往是自己摸索成长,自己不成体系的自学效果低效漫长且无助。
因此收集整理了一份《2024年Java开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。
既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Java开发知识点,不论你是刚入门Java开发的新手,还是希望在技术上不断提升的资深开发者,这些资料都将为你打开新的学习之门!
如果你觉得这些内容对你有帮助,需要这份全套学习资料的朋友可以戳我获取!!
由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!
料的朋友可以戳我获取!!**](https://bbs.csdn.net/topics/618164986)
由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!