新生代调优
新生代的特点
- TLAB thread-local allocation buffer,线程局部缓冲区,线程使用自己私有区域分配对象内存
- 所有的new 操作的内存分配非常廉价
- 死亡对象的回收代价是零;因为采用复制算法,存活的对象使用复制算法到Survivor区域,剩下都是需要被回收的
- 大部分对象用过即死,只有少数对象存活
- Minor GC 的时间远远低于Full GC
- 新生代优化空间更大一些
如何给新生代调优呢?是不是将新生代内存调得越大越好?下面是Oracle官方文档说明截图
网页链接:https://docs.oracle.com/en/java/javase/11/tools/java.html#GUID-3B1CE181-CD30-4178-9602-230B800D4FAE
上述大致中文翻译:设置年轻代的堆的初始大小和最大大小(以字节为单位)。字母k或K表示千字节,m或M表示兆字节,g或G表示千兆字节。堆的年轻代区域用于新对象。与其他区域相比,在该区域执行GC的频率更高。如果年轻代设置太小,则会执行大量 minor gc垃圾回收。如果设置太大,则仅执行full gc垃圾回收才有效,这可能需要很长时间才能完成。Oracle官方建议设置年轻代的大小保持大于堆总大小的25%,并且小于堆总大小的50%。
总结:新生代,还是需要调大一些,因为新生代采用复制算法,需要移动对象,复制算法性能效率较低。
公式:新生代能容纳所有【并发量 * (请求 - 响应)】的数据
- 幸存区大到能保留【当前活跃对象 + 需要晋升对象】,原则就是让真正需要进入老年代的对象才进入老年代。
- 晋升阈值配置得当,让长时间存活对象尽快晋升
-XX:MaxTenuringThreshold=threshold // 设置年龄阈值,大值为15。并行(吞吐量)收集器的默认值为15,而CMS收集器的默认值为6。
-XX:+PrintTenuringDistribution // 启用打印保有权年龄信息,这个参数对于设置-XX:MaxTenuringThreshold有很大帮助,阀值需要长时间观察对象分布,设置合理即可。
老年代调优
以CMS 为例
- CMS的老年代内存越大越好
- 先尝试不做调优,如果没有Full GC 那么老年代已经足够大了;如果有Full GC 则先尝试调优新生代
- 观察发生Full GC 时老年代内存占用,将老年代内存预设调大 1/4 ~ 1/3
-XX:CMSInitiatingOccupancyFraction=percent // 控制老年代占用空间大小占总空间大小比例,进行CMS垃圾回收;值越小就越早进行垃圾回收,推特工程师有一个演讲建议将此值设置为0,即一有垃圾就回收;一般我们将此值设置75%~80%之间,预留25%-20%给浮动垃圾