对象被判定为垃圾的标准
没有被其他对象引用的对象应该被判定为垃圾.
判断对象是否为垃圾的算法
- 引用计数算法
- 可达性分析算法
引用计数算法
- 通过判断对象的引用数量来决定对象是否可以被回收.
- 在这种机制下,堆中的每个实例对象都有一个
引用计数器
,对象被引用则计数器+1,完成引用(引用失效)则-1. - 当引用计数器值为0的时候,该实例对象便被认为可以被当做垃圾回收.
引用计数算法优点:
执行效率高,程序执行受影响较小
引用计数算法缺点:
无法检测出循环引用的情况,可能会因此导致内存泄漏.
比如两个对象互相引用,对象ojbA和对象objB都有字段instance,赋值令objA.instance = objB和objB.instance = objA
,那这两个对象永远都不会回收.
但是实际上 ,jvm没有采取这种方式进行垃圾标记.
可达性分析算法
通过判断对象的引用链是否可达来决定对象是否可以被回收.
可达性分析算法原理:
通过一系列名为GC Roots
的对象作为起始点,从这些节点开始向下搜索,搜索所走过的路径就被称为引用链,当一个对象到GC Roots没有任何引用链相连
的时候,就认为从GC Roots到这个对象是不可达的,也就证明了该对象是不可用的,可以被回收.
可以作为GC Roots的对象:
- 虚拟机栈中引用的对象(栈帧中的本地变量表里面引用的对象),
比如说我们将一个new出来的对象赋值给了一个局部变量,那么在这个方法执行结束前,这个new出来的对象就可以作为GC Root
. - 方法区(
也就是元空间,存放的是类的信息
)中常量引用的对象, 比如在类里定义了一个常量,该常量保存的时某个对象的地址,那么这个对象也是GC Root
. - 方法区中类静态属性引用的对象,和第二条差不多.
- 本地方法栈中JNI(Native方法)的引用对象.
- 活跃状态的线程(也就是线程还在跑的状态下)的引用对象.