JVM+GC解析(题目1)
1、JVM垃圾回收的时候如何确定垃圾?是否知道什么是GC Roots?
GC Root如何确定?那些对象可以作为GC Root?
1.1、什么是垃圾?
简单说就是内存中已经不再被使用到的空间就是垃圾
比如一个对象指向NULL了,慢慢得就会被回收了
1.2、要进行垃圾回收,如何判断一个对象是否可以被回收?
1.2.1、引用计数法
给对象添加一个引用计数器
每当有一个地方引用它,计数器值加1
每当有一个引用失效时,计数器值减1
任何时刻计数器值为0的对象就是不可能再被使用的,那么这个对象就是可回收对象。
但是主流JVM中并没有选用这个算法,因为这种算法很难解决对象之间相互循环引用的问题。
1.2.2、枚举根节点做 可达性分析(根搜索路径)
为了解决引用计数器的循环引用问题,Java使用了可达性分析的方法。
跟踪 (Tracing)
- 复制(Copying)
- 标记清除(Mark-Sweep)
- 标记压缩(Mark-Compact)(Mark-Sweep-Compact)
GC引用遍历
对象B就是对象可达,对象C基本上就称为对象不可达
所谓“GC Roots”或者说Tracing GC的“根集合”就是一组必须活跃的引用。
基本思路就是通过一系列名为“GC Roots”的对象作为起始点,从这个名为“GC Roots”的对象开始向下搜索,如果一个对象到GC Roots没有任何引用链相连时,则说明这个对象不可用(例如上图对象C)。也即给定一个集合的引用作为根出发,通过引用关系遍历对象图,能被遍历到的(可达到的)对象就被判定为存活;没有遍历到的(不可达的)就被判定为死亡。
简单理解:人为设置一些人为贵族,凡是和这些贵族没有血缘关系的人则逐出城外。
那么问题就来到了,到底哪些对象应该被设置为GC Roots”对象呢?
- 虚拟机栈(栈帧中的局部变量区,也叫局部变量表)中引用的对象。
- 方法区中的类静态属性引用的对象。
- 方法区中的常量引用的对象。
本地方法栈中JNI(Native方法)引用的对象。
