运行时数据区域
Java虚拟机在执行Java程序的过程会把它所管理的内存划分为若干个不同的数据区域,Java虚拟机所管理的内存包括以下几个运行时数据区域
程序计数器
程序计数器是一块较小的内存区域,它可以看作是当前线程所执行的字节码的行号指示器。在多线程环境下,每个线程都有独立的程序计数器,互不影响。它保证了线程切换后能够恢复到正确的执行位置,是线程私有的内存区域。
Java虚拟机栈
Java虚拟机栈也是线程私有的,每个方法被执行时都会创建一个栈帧,用于存储方法的局部变量、操作数栈、动态链接、方法出口等信息。在方法执行时,栈帧会被创建并入栈,方法执行完毕后栈帧会出栈。
本地方法栈
本地方法栈与虚拟机栈所发挥的作用是非常相似的,区别在于虚拟机栈为虚拟机执行Java方法(字节码)服务,而本地方法栈则是为虚拟机使用本地(native)方法服务
Java堆
Java堆是Java虚拟机管理的最大的一块内存区域,用于存放对象实例。所有线程共享Java堆,在堆中分配的对象可以被所有线程访问。Java堆是垃圾收集器管理的主要区域,用于存储被创建的对象和数组。
方法区
方法区和Java堆一样,是各个线程共享的内存区域,存储被虚拟机加载的类信息、常量、静态方法、即时编译器编译后的代码缓存等数据。在HotSpot虚拟机中,JDK8以前,方法区也被大部分成为永久区,JDK8以后元空间取代了永久区
运行时常量
运行时常量池是方法区的一部分,Class文件中有一个常量池表,用于存放编译期生成的各种字面量与符号引用,这部分会在类加载后存放到方法区的运行时常量池中
直接内存
直接内存并不是虚拟机运行时数据区的一部分,这部分内存在NIO类可以直接使用Native函数库直接分配堆外内存,避免在Java堆和Native堆中来回复制数据,提高了I/O性能
JVM整体结构及内存模型
JVM内存参数设置
Spring Boot程序的JVM参数设置格式(Tomcat启动直接加在bin目录下catalina.sh文件里):
java ‐Xms2048M ‐Xmx2048M ‐Xmn1024M ‐Xss512K ‐XX:MetaspaceSize=256M ‐XX:MaxMetaspaceSize=256M ‐jar microservice‐eureka‐server.jar
关于元空间的JVM参数有两个:-XX:MetaspaceSize=N和 -XX:MaxMetaspaceSize=N
-XX:MaxMetaspaceSize: 设置元空间最大值, 默认是-1, 即不限制, 或者说只受限于本地内存大小。
-XX:MetaspaceSize: 指定元空间触发Fullgc的初始阈值(元空间无固定初始大小), 以字节为单位,默认是21M,达到该值就会触发
full gc进行类型卸载, 同时收集器会对该值进行调整: 如果释放了大量的空间, 就适当降低该值; 如果释放了很少的空间, 那么在不超
过-XX:MaxMetaspaceSize(如果设置了的话) 的情况下, 适当提高该值。这个跟早期jdk版本的-XX:PermSize参数意思不一样,-
XX:PermSize代表永久代的初始容量。
由于调整元空间的大小需要Full GC,这是非常昂贵的操作,如果应用在启动的时候发生大量Full GC,通常都是由于永久代或元空间发生
了大小调整,基于这种情况,一般建议在JVM参数中将MetaspaceSize和MaxMetaspaceSize设置成一样的值,并设置得比初始值要大,
对于8G物理内存的机器来说,一般我会将这两个值都设置为256M。
结论:
-Xss设置越小count值越小,说明一个线程栈里能分配的栈帧就越少,但是对JVM整体来说能开启的线程数会更多
评论区