侧边栏壁纸
博主头像
兰若春夏 博主等级

一日为坤,终生为坤

  • 累计撰写 19 篇文章
  • 累计创建 12 个标签
  • 累计收到 4 条评论

目 录CONTENT

文章目录

运行时数据区域

奥德坤
2024-10-01 / 0 评论 / 1 点赞 / 51 阅读 / 0 字

运行时数据区域

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整体来说能开启的线程数会更多

1

评论区