引用计数法:
程序给对象添加一个引用计数器,每有一个变量引用它时计数器加1,当引用断开时计数器减一。当计数器为0时,代表没有任何变量引用他,该对象就死亡状态,JVM需要对此类对象进行回收。此计数法无法回收具有循环引用的对象,由于对象相互引用,导致各自计数器都不为0,导致JVM无法回收。
可达性分析法:
程序由GC Roots作为起点,从这些节点开始向下搜索,搜索走过的路径称为引用链,当一个对象到GC Roots没有任何引用链时,证明此对向是不可用的。 Java中,作为GC Roots的对象有一下几种:
- 虚拟机栈(栈帧中的本地变量表)中的引用对象
- 方法区中类静态属性引用的对象
- 方法区中常量引用的对象
- 本地方法栈(native方法)中引用的对象
/**
* 可作为GC Roots的对象
* 1、虚拟机栈(栈帧中的本地变量表)中引用的对象
* 2、方法区中类静态属性引用的对象
* 3、方法区中常量引用的对象
* 4、本地方法栈中JNI(Native方法)引用的对象(和虚拟机栈类似,一个是虚拟机层面的调用,一个是本地层面的调用)
**/
public class Demo1 {
/**
* 1、虚拟机栈(栈帧中的本地变量表)中引用的对象
* 此时的 s,即为 GC Root,当s置空时,parameter 对象也断掉了与 GC Root 的引用链,将被回收
*/
public static void testGC1() {
StackLocalParameter s = new StackLocalParameter("parameter");
s = null;
}
/**
* 2、方法区中类静态属性引用的对象
* s 为 GC Root,s 置为 null,经过 GC 后,s 所指向的 staticProperties 对象由于无法与 GC Root 建立关系被回收。
* 而 m 作为类的静态属性,也属于 GC Root,parameter 对象依然与 GC root 建立着连接,所以此时 parameter 对象并不会被回收。
*/
public static void testGC2() {
MethodAreaStaicProperties s = new MethodAreaStaicProperties("staticProperties");
s.m = new MethodAreaStaicProperties("parameter");
s = null;
}
/**
* 3、方法区中常量引用的对象
* m 即为方法区中的常量引用,也为 GC Root,s 置为 null 后,final 对象也不会因没有与 GC Root 建立联系而被回收
*/
public static void testGC3() {
MethodAreaConstant s = new MethodAreaConstant("parameter");
s = null;
}
}
/**
* 1、虚拟机栈
*/
class StackLocalParameter {
public StackLocalParameter(String name) {
}
}
/**
* 2、方法区静态属性
*/
class MethodAreaStaicProperties {
public static MethodAreaStaicProperties m;
public MethodAreaStaicProperties(String name) {
}
}
/**
* 3、方法区中常量引用的对象
*/
class MethodAreaConstant {
public static final MethodAreaConstant m = new MethodAreaConstant("final");
public MethodAreaConstant(String name) {
}
}