时隔这么久,这个系列又和大家见面了,感谢支持我的朋友,只要能给大家带来一点点收获,我真的时发自内心的开心!
今天先简单了解一下虚拟机的垃圾收集器。
前面已经介绍过了,程序计数器,虚拟机栈和本地方法栈是属于线程私有区域,这三个区域分配和回收都是确定的,方法和线程结束后,内存自认而然就回收了。
而Java堆和方法作为公共区域回收则比较复杂,垃圾收集器所关注也是这两个部分的内存。
但是如何判断哪些对象应该被回收呢?
比较简单的是引用计数算法:给对象增加一个计数器,多一个地方引用此对象,该对象的计数器就加1;少一个对此对象的引用,计数器就减1。
这种方式的优点是判断效率高,但是也有一个严重的缺点:存在对象之间循环引用的问题。
比如在一段程序中,有如下引用方式,objA.instacce=objB和objB.instacce=objA,这种将不会被回收。
所以主流的Java虚拟机里面并没有使用这种方式,而是采用了另外一种方式——可达性分析算法。
基本思想:通过一系列的称为“GC Roots”的对象作为起始点,从这些节点开始向下搜索,搜索所 走过的路径称为引用链(Reference Chain),当一个对象到GC Roots没有任何引用链相连 (用图论的话来说,就是从GC Roots到这个对象不可达)时,则证明此对象是不可用的。(摘自《深入理解JAVA虚拟机》)
如下图所示,object5,object6, object7没有道GC Root对象的引用链,是可以被回收的。注意,这里是可以被回收,这时并不一定会被立刻回收。
那么,什么样的对象可以被选作为GC Roots对象呢,主要有以下四类。
既然提到了”引用”,那么JAVA中引用是怎么定义的呢?
传统的定义(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类来创建,不会对垃圾回收构成任何影响,用处是可以在被回收时接到一个系统通知。。
对象死亡会经历两次标记的过程,流程如下图所示。
回收方法区
1 在新生代中,回收率效率,很高可以达到70%到95%的空间
2 在永久代中,主要回收废弃常量和无用的类
“无用的类”的判定条件:
1. 该类所有的实例已经被回收
2. 加载该类的ClassLoder已经被回收
3. 该类对应的java.lang.Class对象没有任何对方被引用