深入理解JAVA虚拟机学习笔记4——垃圾收集器概述

时隔这么久,这个系列又和大家见面了,感谢支持我的朋友,只要能给大家带来一点点收获,我真的时发自内心的开心!

今天先简单了解一下虚拟机的垃圾收集器。

前面已经介绍过了,程序计数器,虚拟机栈和本地方法栈是属于线程私有区域,这三个区域分配和回收都是确定的,方法和线程结束后,内存自认而然就回收了。

而Java堆和方法作为公共区域回收则比较复杂,垃圾收集器所关注也是这两个部分的内存。

但是如何判断哪些对象应该被回收呢?

比较简单的是引用计数算法:给对象增加一个计数器,多一个地方引用此对象,该对象的计数器就加1;少一个对此对象的引用,计数器就减1。

这种方式的优点是判断效率高,但是也有一个严重的缺点:存在对象之间循环引用的问题。

比如在一段程序中,有如下引用方式,objA.instacce=objB和objB.instacce=objA,这种将不会被回收。

所以主流的Java虚拟机里面并没有使用这种方式,而是采用了另外一种方式——可达性分析算法。

基本思想:通过一系列的称为“GC Roots”的对象作为起始点,从这些节点开始向下搜索,搜索所 走过的路径称为引用链(Reference Chain),当一个对象到GC Roots没有任何引用链相连 (用图论的话来说,就是从GC Roots到这个对象不可达)时,则证明此对象是不可用的。(摘自《深入理解JAVA虚拟机》)

如下图所示,object5,object6, object7没有道GC Root对象的引用链,是可以被回收的。注意,这里是可以被回收,这时并不一定会被立刻回收。

深入理解JAVA虚拟机学习笔记4——垃圾收集器概述_第1张图片

那么,什么样的对象可以被选作为GC Roots对象呢,主要有以下四类。

既然提到了”引用”,那么JAVA中引用是怎么定义的呢?

深入理解JAVA虚拟机学习笔记4——垃圾收集器概述_第2张图片

传统的定义(1.2之前):如果reference类型的数据中存储的数值代表的是另外一块内存的起始地址。

但是在1.2之后,JAVA对引用进行了扩充,又分出以下四种(这里不知道大家有没有和我一开始一样的困惑,这个引用类型不是说程序执行之后动态产生的引用状态,而是在我们开发的时候,自己手动设置的引用类型)。

1 强引用:类似Object o = new Object()这类的引用,只要引用还在,即使出现OutOfMemoryError也不会被回收。

2 软引用:使用WeakReference类来创建,代码如下,当a=null的时候,垃圾收集器将会把a列入回收范围,在内存不足时进行回收。属于有用但非必需的对象。

A a = new A();

SoftReference b = new SoftReference (a);

3 弱引用:使用WeakReference类来创建,代码如下,当a=null的时候,垃圾收集器将会把a列入回收范围,垃圾收集的时候会被立刻回收。属于非必需的对象。

4 虚引用:使用PhantomReference类来创建,不会对垃圾回收构成任何影响,用处是可以在被回收时接到一个系统通知。。

深入理解JAVA虚拟机学习笔记4——垃圾收集器概述_第3张图片

对象死亡会经历两次标记的过程,流程如下图所示。

回收方法区

深入理解JAVA虚拟机学习笔记4——垃圾收集器概述_第4张图片

1 在新生代中,回收率效率,很高可以达到70%到95%的空间

2 在永久代中,主要回收废弃常量和无用的类

“无用的类”的判定条件:

1. 该类所有的实例已经被回收

 2. 加载该类的ClassLoder已经被回收

 3. 该类对应的java.lang.Class对象没有任何对方被引用

你可能感兴趣的:(深入理解JAVA虚拟机学习笔记4——垃圾收集器概述)