1.调整堆的大小
1.1.随着堆的大小增加,停顿的持续时间也会增加
1.2.停顿发生的频率确实会降低,但是停顿的持续时间会拖慢整体性能
1.3.操作系统进行的交换对JVM是不公开的
1.3.1.操作系统要将数据从磁盘交换到RAM,这是一个代价高昂的操作
1.4.首要规则是设定堆的大小永远不要超过机器的物理内存
1.4.1.如果有多个JVM在运行,那么这适用于所有堆的总和
1.5.-XmsN标志
1.5.1.初始值
1.6.-XmxN标志
1.6.1.最大值
1.7.默认值
1.7.2.在物理内存小于192 MB的机器上,堆的最大值会是物理内存的一半(96 MB或更少)
1.8.如果JVM发现堆在初始大小时,GC的次数太多,它就会不断地增加堆大小,直到JVM执行的GC数量“适当”,或者直到堆大小达到最大值
1.9.对于不需要堆很大的应用程序来说,根本不需要设置堆的大小,只需要设定GC算法的性能目标
1.10.除非应用程序需要比默认值更大的堆,否则优化时可以调整GC算法的性能目标而不是微调堆的大小
1.11.一个有效的经验是,调整堆的大小,让其在Full GC之后,仍然被占用30%
1.11.1.用jconsole连接应用程序,强制执行Full GC,并观察当Full GC完成后有多少内存被占用
1.11.2.确保它有额外的0.5 GB~1 GB内存来满足JVM的非堆需求
1.12.当JVM运行在隔离容器中时,你需要设定堆的最大值
1.13.确切地知道应用程序需要多大的堆,那么不妨将初始值和最大值都设置为该值
1.13.1.因为GC不再需要弄清楚是否应该调整堆大小,稍微高效一点
2.调整分代大小
2.1.新生代相对较大
2.1.1.Young GC的停顿时间会增加
2.1.2.新生代的回收频率会降低
2.1.3.晋升到老年代的对象也会更少
2.1.4.老年代相对会更小
2.1.4.1.被填满的频率会更高
2.1.4.2.会执行更多的Full GC
2.2.优化分代大小的命令行标志,调整的都是新生代的大小,老年代自动得到剩余的所有空间
2.3.-XX:NewRatio=N
2.3.1.设置新生代与老年代的比例
2.3.2.新生代初始大小=堆的初始大小/(1 +NewRatio)
2.3.3.默认值是2
2.3.3.1.新生代的初始大小是堆初始大小的33%
2.3.4.如果应用程序会动态调整堆大小,并且需要更大的(或更小的)新生代,那就重点设置NewRatio的值
2.4.-XX:NewSize=N
2.4.1.设置新生代的初始值
2.4.2.设置了该选项,那么它将优先于通过NewRatio计算出来的值
2.4.3.没有默认值
2.4.3.1.默认情况下这个值会通过NewRatio计算出来
2.5.-XX:MaxNewSize=N
2.5.1.设置新生代的最大值
2.5.2.通过设定新生代的最小值和最大值来进行优化是十分困难的
2.6.-XmnN
2.6.1.将NewSize和MaxNewSize设置为同一个值的简单写法
2.6.2.如果堆的大小是固定的(通过设置-Xms等于-Xmx),通常最好也用-Xmn将新生代大小设为固定的
2.7.自适应大小
2.7.1.一般应该保持开启
2.7.1.1.因为GC算法会通过调整代的大小,来尽量满足它们停顿时间的目标
2.7.1.2.控制着JVM改变堆内新生代和老年代比例的方式
2.7.2.关闭-XX:-UseAdaptiveSizePolicy标志
2.7.2.1.默认值为true
2.7.3.如果堆的最小值和最大值设置为相同的值,并且新生代的初始值和最大值也设为相同的值,那么自适应大小实际上就被关闭了
2.7.4.如果你想将其中某个阶段的GC优化到最佳状态,禁用自适应大小很有用
2.7.5.对于精细优化过的堆,禁用自适应大小能小幅提升性能
2.7.6.开启-XX:+PrintAdaptiveSizePolicy标志
2.7.6.1.查看JVM是如何调整应用程序内空间大小
2.7.6.2.执行GC时,GC日志里会包含回收过程中各个代是如何调整大小的详细信息