如果有错希望指出。本文是在看到一些关于JVM参数调优文章后的一些内容摘要。
堆大小设置
-Xms
设置JVM 初始内存,即JVM启动时分配的内存。此值可以设置与 -Xmx 相同,以避免每次垃圾回收完成后 JVM 重新分配内存。
-Xmx
设置JVM 运行过程中分配的最大可用内存。
-Xss
-Xss128k:设置每个线程的堆栈大小。
-Xmn
设置年轻代大小。
打印 GC
- -XX:+PrintGC:打印 GC 信息
- -XX:+PrintGCDetails
- -XX:PrintHeapAtGC:打印GC前后的详细堆栈信息
-XX:+AlwaysPreTouch
JAVA进程启动的时候,虽然我们可以为JVM指定合适的内存大小,但是这些内存操作系统并没有真正的分配给JVM,而是等JVM访问这些内存的时候,才真正分配,这样会造成以下问题:
- 第1次YGC之前Eden区分配对象的速度较慢;
- YGC的时候,Young区的对象要晋升到Old区的时候,这个时候需要操作系统真正分配内存,这样就会加大YGC的停顿时间;
AlwaysPreTouch
参数可以优化上面的问题,但是它的副作用会导致 JVM 进程启动时间变长。
启动时间变长原因
在没有配置-XX:+AlwaysPreTouch参数即默认情况下,JVM参数-Xms申明的堆只是在虚拟内存中分配,而不是在物理内存中分配:它被以一种内部数据结构的形式记录,从而避免被其他进程使用这些内存。这些内存页直到被访问时,才会在物理内存中分配。当JVM需要内存的时候,操作系统将根据需要分配内存页。
配置-XX:+AlwaysPreTouch参数后,JVM将-Xms指定的堆内存中每个字节都写入’0’,这样的话,除了在虚拟内存中以内部数据结构保留之外,还会在物理内存中分配。并且由于touch这个行为是单线程的,因此它将会让JVM进程启动变慢。所以,要么选择减少接下来对每个缓存页的第一次访问时间,要么选择减少JVM进程启动时间,这是一种trade-off。