(JVM学习02)JVM内存模型深度剖析与优化

一、JDK体系结构

在这里插入图片描述

二、Java语言的跨平台特性

在这里插入图片描述

三、JVM整体结构及内存模型

以JDK8为例
在这里插入图片描述

JVM虚拟机

1.类装载子系统

c++实现的,负责加载class文件到jvm内存(运行时数据区)

2.运行时数据区
1)堆
年轻代

Eden:小对象初始分配在该区域中
Survivor:分为s0,s1;当minor gc时,不能回收的对象一般会进入s0或s1;同一时间内s0和s1只有一个存有对象,另一个是空的,比如gc时s0有对象,就会将s0不能回收的对象转移到s1,然后清空s0;在Survivor中每次gc不能回收的对象分代年龄会+1,达到一定次数(最大15次)就进入到老年代。

老年代

储存大对象、不能被回收的对象、分代年龄高的对象

2)方法区(元空间)

储存常量、静态变量(只是存静态的对象在堆空间的地址)、类信息

3)栈

线程私有,该线程每执行一个方法,就会往栈中插入一个栈帧。因为后执行的方法需要先出栈被销毁,所以使用栈的结构

栈帧
  1. 局部变量表:存储局部变量
  2. 操作数栈:操作数据的中转空间,也是栈结构
  3. 动态链接:在程序运行期间完成的将符号引用替换为直接引用的地址
  4. 方法出口(返回地址):储存该方法执行完后返回到之前哪个位置的信息
4)本地方法栈

线程私有,jvm调用本地方法(native)时分配的内存区域。虚拟机可以自由实现它,在HotSopt中直接就把本地方法栈和栈合二为一

5)程序计数器

线程私有,储存正在执行的的代码地址行号

3.字节码执行引擎

c++实现的,负责执行运行时数据区的代码和修改程序计数器

四、JVM内存参数设置

在这里插入图片描述
-Xms: 初始堆内存大小,如果数据超过该大小可逐步扩容至-Xmx大小
-Xmx: 最大可分配堆内存大小,如果超过将会OOM
-Xmn: 新生代内存大小
-Xss: 每个线程可使用的内存大小,即栈的大小

关于元空间的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。

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

 java ‐Xms2048MXmx2048MXmn1024MXss512K ‐XX:MetaspaceSize=256M ‐XX:MaxMetaspaceSize=256M ‐jar microservice‐eureka‐server.jar
StackOverflowError示例
1 // JVM设置 ‐Xss128k(默认1M)
2 public class StackOverflowTest {
3
4 static int count = 0;
5
6 static void redo() {
7 count++;
8 redo();
9 }
10
11 public static void main(String[] args) {
12 try {
13 redo();
14 } catch (Throwable t) {
15 t.printStackTrace();
16 System.out.println(count);
17 }
18 }
19 }
20
21 运行结果:
22 java.lang.StackOverflowError
23 at com.tuling.jvm.StackOverflowTest.redo(StackOverflowTest.java:12)
24 at com.tuling.jvm.StackOverflowTest.redo(StackOverflowTest.java:13)
25 at com.tuling.jvm.StackOverflowTest.redo(StackOverflowTest.java:13)
26 ......

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

日均百万级订单交易系统如何设置JVM参数

在这里插入图片描述
结论:通过上面这些内容介绍,大家应该对JVM优化有些概念了,就是尽可能让对象都在新生代里分配和回收,尽量别让太多对象频繁进入老年代,避免频繁对老年代进行垃圾回收,同时给系统充足的内存大小,避免新生代频繁的进行垃圾回收。

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值