JVM day03
如何判断对象可以回收
引用计数法
- 一旦相互持有引用,就导致对象永远没法被回收。
- jvm并没有采用这种算法。
可达性分析算法
- 由GC Root出发,开始寻找,看看某个对象是否可达,如果可到达则不回收,不可到达则当做垃圾回收
可以作为根对象的有哪些?
- 类加载器、Thread、本地变量表、static成员、常用引用、局部变量引用的对象、本地方法栈中的变量等
四种引用
- 实线指强引用
- 强引用
指创建一个对象并把这个对象赋给一个引用变量。强引用有引用变量指向时永远不会被垃圾回收,JVM宁愿抛出OutOfMemory错误也不会回收这种对象。
- 软引用
如果一个对象具有软引用,内存空间足够,垃圾回收器就不会回收它;
**如果内存空间不足了,就会回收这些对象的内存。只要垃圾回收器没有回收它,该对象就可以被程序使用。**软引用可用来实现内存敏感的高速缓存,比如网页缓存、图片缓存等。使用软引用能防止内存泄露,增强程序的健壮性。
当软引用引用的A2对象被回收时,软引用对象会进入引用队列,方便进一步对其内存释放,弱引用同理
- 应用(强引用时,某些不重要的对象因为被强引用导致无法被回收,可以使用软引用来让其被回收掉)
- 清理软引用(使用引用队列)
3. 弱引用
弱引用也是用来描述非必需对象的,当JVM进行垃圾回收时,无论内存是否充足,都会回收被弱引用关联的对象。
4 .虚引用
虚引用和前面的软引用、弱引用不同,它并不影响对象的生命周期。如果一个对象与虚引用关联,则跟没有引用与之关联一样,在任何时候都可能被垃圾回收器回收。主要配合ByteBuffer使用。
要注意的是,虚引用必须和引用队列关联使用,当垃圾回收器准备回收一个对象时,如果发现它还有虚引用,就会把这个虚引用加入到与之 关联的引用队列中。程序可以通过判断引用队列中是否已经加入了虚引用,来了解被引用的对象是否将要被垃圾回收。如果程序发现某个虚引用已经被加入到引用队列,那么就可以在所引用的对象的内存被回收之前采取必要的行动。
垃圾回收算法
标记清除
先标记后清除
- 优点:速度快
- 缺点:造成的内存碎片变多
标记整理
先标记后整理
- 优点:不会造成内存碎片
- 缺点:速度较慢
复制
将可用内存按容量划分为大小相等的两块,每次只用其中一块。当这块内存用完了,就将还存活的对象复制到另外一块上面,然后再把已使用过的内存空间一次清理掉。
- 优点:相比于标记清除算法,解决了不连续空间碎片的问题
- 缺点:将空间一分为二,浪费一半的空间
分代垃圾回收
- 新生代
一般占据堆的 1/3 空间。用来存放新生的对象,新生代对象朝生夕死,对象存活率低,新生代MinorGC非常频繁,一般回收速度也比较快。新生代常采用 复制算法。
- 老年代
主要存放应用程序中生命周期长的内存对象。老年代对象存活率高。老年代MajorGC经常会伴随至少一次的 MinorGC,老年代MajorGC 的速度比较慢。老年代常采用的 标记-清除算法和标记-整理算法。
- 对象首先分配在伊甸园区域
- 新生代空间不足时,触发minor gc,伊甸园和from存活的对象使用copy复制到to中,存活的对象
年龄加一并且交换from区和to区,其最大寿命是15(4bit)。当对象寿命超过阈值时,会晋升至老年代。
-
minor gc会引发stop the world(STW),暂停其他用户的线程,等垃圾回收结束用户线程才恢复运行。
-
当老年代空间不足,会先尝试触发minor gc,如果之后空间仍不足,那么触发full gc,其stop the world的时间更长(STW)