重温深入理解java虚拟机这本书,温故而知新。
书是基于java虚拟机规范而来,文章中会掺杂我的个人理解的描述,如有误,请指正。
1、对象收集相关算法
(1)引用计数算法(虚拟机没有采用此种算法,无法解决循环依赖的问题)
(2)可达性分析算法
当一个对象到GC Roots没有任何引用链,则为可回收的对象
GC Roots的对象包括:虚拟机栈中引用的对象、方法区中静态属性应用的对象、方法去中常量引用的对象、本地方法栈中的JNI引用的对象。
(3)四种引用:强引用 软引用 弱引用 虚引用
(4)方法区的回收
2、垃圾收集算法
(1)标记-清除算法
标记可回收对象,清理被标记的对象,这两步效率不高,会产生内存碎片。
老年代可以采用此种方式
(2)复制算法
将房子一分为二,一半放杂物,过一段时间后你发现放东西的一半满了,于是你就把不用的杂物扔掉了,然后剩下的杂物搬到屋子的另一边,搬的过程顺手整理一下,这样开始放杂物的一半就空出来,可以放更多的东西了。
因为1:1这种方式浪费空间,所以现在的虚拟机采用Eden空间和两块较小的Survivor空间,HotSpot默认Eden和Survivor比例为8:1,当三块地都不够用的时候,老年代就可以用上了。
新生代采用此种方式
(3)标记-整理算法
当对象存活率较高时候,会出现较多的复制操作,效率变低,而且会浪费额外的空间,基于以上问题,老年代可以采用标记可回收对象,移动存活的对象向一边移动,清理边界之外的内存的方式。与(1)方式不同点在于不直接清理,而是移动存活对象再进行清理。
基于以上算法介绍,虚拟机采用了分代收集的算法应对不同代的回收,原因是新生代中,每次收集会有大批量的对象死去,少量存活,复制算法效率高一些。老年代相反,没有额外的空间进行分配担保,必须使用(1)和(3)两种算法进行。
3、HotSpot的算法实现
(1)虚拟机发起内存回收的过程
A、枚举根节点
可达性分析中从GCRoots节点找引用链,如果逐个检查会消耗非常多的时间,并且在进行分析的时间段中引用关系不能变化,所以会导致stop the world。基于以上原因,hotspot采用了一组OopMap的数据结构存放引用。
B、安全点(Safepoint)
关于安全点,我也没有完全理解透彻,附上一篇文章说明下:https://www.jianshu.com/p/c79c5e02ebe6
为了解决OopMap内容变化的指令非常多,会产生大量的额外空间,所以只在特定的位置记录信息
C、安全区域
当线程处于Sleep阶段或者Blocked状态,这样就不能进入安全点,这时可以通过标定一块安全区域,安全区域内的代码不会改变引用关系,可以放心的进行GC,离开安全区域时,要进行Safepoint的检查
(2)垃圾收集器(即具体怎么回收的)
http://www.cnblogs.com/redcreen/archive/2011/05/04/2037057.html
A、Serial收集器(新生代收集器,复制算法)
垃圾收集时必须暂停其他所有的工作线程,直到收集结束,没有线程切换的消耗,在client模式下仍然是默认的垃圾收集器的实现。可以与CMS配合使用。
设置参数:
"-XX:+UseSerialGC":添加该参数来显式的使用串行垃圾收集器;
B、ParNew收集器(新生代,多线程版本)
可以与CMS配合使用。
设置参数:
"-XX:+UseConcMarkSweepGC":指定使用CMS后,会默认使用ParNew作为新生代收集器;
"-XX:+UseParNewGC":强制指定使用ParNew;
"-XX:ParallelGCThreads":指定垃圾收集的线程数量,ParNew默认开启的收集线程与CPU的数量相同;
C、Parallel Scavenge 收集器(新生代收集器,关注可控的吞吐量)
吞吐量 = 运行用户代码时间/(运行用户代码时间+垃圾收集时间),例如总共运行100分钟,垃圾收集花掉1分钟,吞吐量为99%。
设置参数:
"-XX:MaxGCPauseMillis":控制最大垃圾收集停顿时间
"-XX:GCTimeRatio":直接设置吞吐量
"-XX:+UseAdaptiveSizePolicy":开关参数,打开之后可以自动调节堆内存的代分配比例达到最优,称为GC的自适应的调节策略,也是与ParNew收集器的一个重要区别。
D、Serial Old收集器(老年代收集器,标记-整理算法)
单线程的,在1.5之前跟Parallel Scavenge配合使用,在CMS发生Concurrent Mode failure时使用。
E、Parallel Old收集器(老年代收集器,标记-整理算法)
与Parallel Scavenge配合的老年代收集器,1.6版本之后可用。
注重吞吐量和cpu资源敏感的场合使用。
F、CMS收集器(老年代收集器,标记-清除算法)
G、G1收集器
4、内存分配与回收策略
(1)对象优先在Eden区分配
例子:-Xms20M -Xmx20M -Xmn10M -XX:+PrintGCDetails -XX:SurvivorRatio=8
public static void main(String[] args){ byte[] a1,a2,a3,a4; int _1MB = 1048576; a1 = new byte[2 * _1MB]; a2 = new byte[2 * _1MB]; a3 = new byte[2 * _1MB]; a4 = new byte[5 * _1MB];}
发生一次MinorGC
(2)大对象直接进入老年代
例子:-XX:PretenureSizeThreshold=3145728 (不能直接写3MB)
(3)长期存活的对象将进入老年代
虚拟机给每个对象定义一个年龄(age)计数器,如果在Eden出生并经过MinorGC仍然存活且能被Survivor容纳的话,将被移动到Survivor中,年龄设为1,在Survivor每熬过一次MinorGC,年龄增加一岁,当到一定年龄(-XX:MaxTenuringThreshold 设置此年龄),就可以晋升老年代。
(4)动态对象年龄判定
提前进入老年代的情况:如果Survivor中相同年龄的所有对象大小总和大于Survivor空间的一般,年龄大于或等于该年龄的对象直接进入老年代。