JVM整体结构及内存模型
-
类装载子系统: 加载 Math 类到 JVM 内存。
-
堆: 存放对象。
-
方法区: 存放常量,静态变量,类信息,静态变量的值是一个对象实例时,方法区存放的是对象在内存的地址。方法区直接占用物理内存。
-
栈: 存放局部变量。每执行一个方法,都会在栈中为这个方法开辟一块空间,用来存放这个方法中的局部变量。
-
程序计数器: 记录每个线程将要执行哪行代码。
栈,本地方法栈,程序计数器都是线程私有的,堆和方法区是线程共享的。
栈的结构
-
局部变量表: 存放方法内的局部变量,如果变量的值是一个对象实例,则存放对象在堆中的地址。局部变量表类似 Table 结构。
-
操作数栈: 存放变量的值,会有指令把栈中的值存入局部变量表中。
-
动态链接: 将普通方法的符号引用转化为直接引用。
-
方法出口: 记录当前方法结束后,调用此方法的方法将要执行哪行代码。图片中 main 方法调用了 compute 方法,所以 compute 的方法出口保存着 main 方法将要执行哪行代码。
JVM内存参数设置
- Xms:堆的初始大小
- Xmx:堆的最大大小
- Xmn:堆的新生代大小
- XX:MaxMetaspaceSize: 设置方法区最大值, 默认是-1, 即不限制, 或者说只受限于本地内存大小。
- XX:MetaspaceSize: 指定方法区触发 Full GC 的初始阈值(元空间无固定初始大小), 以字节为单位,默认是21M左右。
为方法区设定这两个值可以避免频繁触发 Full GC。
- Xss: 为一个线程分配栈空间大小。
如果把这个值设置的太小,容易发生 StackOverflowError。
因为 -Xss 设置越小,意味着一个线程栈里能分配的栈帧就越少,意味着方法调用深度也会变浅。
一个线程能调用的方法深度 ≈ -Xss / 单个栈帧大小