如何选择垃圾收集器&安全点&安全区
如何选择垃圾收集器
优先调整堆的大小
让服务器自己来选择- 如果内存小于100M,使用串行收集器
- 如果是单核,并且没有停顿时间的要求,串行或JVM自己选择
- 如果允许停顿时间超过1秒,选择并行或者JVM自己选
- 如果响应时间最重要,并且不能超过1秒,使用并发收集器
4G以下可以用Parallel,4-8G可以用ParNew+CMS,8G以上可以用G1,几百G以上用ZGC
安全点
- 安全点就是指代码中一些特定的位置,当线程运行到这些位置时它的状态是确定的,这样JVM就可以安全的进行一些操作,比如GC等,所以GC不是想什么时候做就立即触发的,是需要等待所有线程运行到安全点后才能触发
安全点是对正在执行的线程设定的
问题
: 收集器在做GC的时候想执行就执行吗- 并不是,如果这样做可能会打破代码中原子操作,需要在代码上设置安全点的标识位,当线程达到这个安全点时,用户线程会被挂起,然后去执行GC操作,一般安全点有如下几个场景
- 方法返回之前
- 调用某个方法之后
- 抛出异常的位置
- 循环的末尾
- 并不是,如果这样做可能会打破代码中原子操作,需要在代码上设置安全点的标识位,当线程达到这个安全点时,用户线程会被挂起,然后去执行GC操作,一般安全点有如下几个场景
问题
: 是否可以在每行代码都设置一个安全点- 不能说一定不可以,但这样设置相当于在代码执行中
时时刻刻都要去遍历安全点,性能会很差
,但是如果只是在这几种特殊场景,对性能对的影响反而不大,而且容易判定是否能进行GC
- 不能说一定不可以,但这样设置相当于在代码执行中
安全区
- 安全区指的是
一段代码中,引用关系不会发生变化
,这个区域内的任意地方开始GC都是安全
的 - 如果一个线程处于睡眠或中断状态,它就不能响应 JVM 的中断请求,再运行到安全点上,因此 JVM 引入了安全区