分享前抛出几个问题?
▪ 如何判断一个对象是否存活着?
▪为什么java堆内存要分代?
▪一个对象从创建到被回收经历了哪些过程,如果是很大的对象呢?
▪哪些情况会报outofMemoryError,StackOverFlowError?
▪Jvm调优的配置参数,你都知道么?
一、对象已经死了么?
▪引用计数法?
▪可达性分析?
引用计数法
对象有被引用,则引用计数器加1
通过如下示例判断jvm是否是使用的是引用计数法?,如下示例是两个对象互相引用这对方,并且无其他引用,如果是使用引用计数法,那么将不会被回收,如果不是,则可以被回收
/**
* @author hezg
* 测试对象相互引用是否可以被回收
*/
public class ReferenceCountingGC {
public Object instance = null;
private static final int _1MB = 1024 * 1024;
private byte[] bigSize = new byte[2 * _1MB];
public static void main(String[] args){
ReferenceCountingGC objA = new ReferenceCountingGC();
ReferenceCountingGC objB = new ReferenceCountingGC();
objA.instance = objB;
objB.instance = objA;
objA = null;
objB = null;
System.gc();
}
}
运行结果:
从图示[PSYoungGen: 8031K->798K(76288K)] 说明两个对象示例都是有被回收的。显然jvm不是使用引用计数法回收对象。
可达性分析
jvm使用可达性分析算法来判断对象是否存活。
这个算法是通过一系列称为"GCRoot"的对象作为起始点,从这些节点开始向下搜索,搜索所走过的路径称为引用链,当一个对象到GC Root是没有任何引用链相连接,则说明此对象是不可用的,如图:
图中可以很明显的知道,对象1-7到GCRoot是可达的,即对象还存活着,对象8-11到GCRoot是不可达的,即对象已死。
那么什么样的对象可以作为GCRoot呢?有如下四种
▪虚拟机栈(栈帧中的本地变量表)中引用的对象。
▪方法区中类静态属性引用的对象。
▪方法区中常量引用的对象。
▪本地方法栈中JNI(即一般说的Native方法)引用的对象。