JVM内存参数设置

JVM整体结构及内存模型

JVM整体结构及内存模型

JVM内存参数设置

运行时数据区域大小参数设置

Spring Boot程序的JVM参数设置格式(Tomcat启动直接加在bin目录下catalina.sh文件里):

1
java ‐Xms2048M ‐Xmx2048M ‐Xmn1024M ‐Xss512K ‐XX:MetaspaceSize=256M ‐XX:MaxMetaspaceSize=256M ‐jar test‐server.jar

-XX:MaxMetaspaceSize: 设置元空间最大值默认是-1,即不限制,或者说只受限于本地内存大小
-XX:MetaspaceSize: 指定元空间触发Full gc初始阈值(元空间无固定初始大小),以字节为单位,默认是21M,达到该值就会触发full gc进行类型卸载,同时收集器会对该值进行调整:如果释放了大量的空间,就适当降低该值;如果释放了很少的空间,在不超过设置的-XX:MaxMetaspaceSize的值的情况下,适当提高该值。这个跟早期jdk版本的-XX:PermSize参数意思不一样,-XX:PermSize代表永久代的初始容量。
‐Xms:初始堆内存
-Xmx:最大堆内存
-XX:+HeapDumpOnOutofMemoryError:内存异常打印dump
-Xmn:新生代内存
-Xss:每个栈内存大小,默认为1M
-XX:SurvivorRatio=8:新生代内存分配比例(8:1:1)

:由于调整元空间的大小需要Full GC,这是非常昂贵的操作,如果应用在启动的时候发生大量Full GC,通常都是由于永久代或元空间发生了大小调整,基于这种情况,一般建议在JVM参数中将MetaspaceSizeMaxMetaspaceSize设置成一样的值,并设置得比初始值要大,对于8G物理内存的机器来说,一般我会将这两个值都设置为256M

StackOverflowError示例

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
/**
* JVM设置
* -Xss128k
* -Xss默认1M
*/
public class StackOverflowTest {
static int count = 0;

static void redo() {
count++;
redo();
}

public static void main(String[] args) {
try {
redo();
} catch (Throwable e) {
e.printStackTrace();
System.out.println(count);
}
}
}

运行结果:

1
2
3
4
5
6
java.lang.StackOverflowError
at com.eleven.icode.jvm.StackOverflowTest.redo(StackOverflowTest.java:15)
at com.eleven.icode.jvm.StackOverflowTest.redo(StackOverflowTest.java:15)
at com.eleven.icode.jvm.StackOverflowTest.redo(StackOverflowTest.java:15)
……
19790

当-Xss设置越小count值越小,说明一个线程栈里能分配的栈帧就越少,但是对JVM整体来说能开启的线程数会更多。

尽可能让对象都在新生代里分配和回收,尽量别让太多对象频繁进入老年代,避免频繁对老年代进行垃圾回收,同时给系统充足的内存大小,避免新生代频繁的进行垃圾回收