《JVM》第二篇 JVM内存模型深度剖析与优化

news2025/1/11 14:08:28

目录

      • 一. JDK体系结构与跨平台特性介绍
      • 二. JVM内存模型深度剖析
      • 三. 从Jvisualvm来研究下对象内存流转模型
      • 四. GC Root与STW机制
      • 五. JVM参数设置通用模型

一. JDK体系结构与跨平台特性介绍

在这里插入图片描述

二. JVM内存模型深度剖析

在这里插入图片描述

  1. 按照线程是否共享来划分
    在这里插入图片描述
    TLAB(Thread Local Allocation Buffer)线程本地分配缓存区,这是一个线程专用的内存分配区域
    由于对象一般会分配在上,而堆是全局共享的。因此在同一时间,可能会有多个线程在堆上申请空间。因此,每次对象分配都必须要进行同步,在竞争激烈的场合分配的效率又会进一步下降, JVM使用TLAB来避免多线程冲突,在给对象分配内存时,每个线程使用自己的TLAB,这样可以避免线程同步,提高了对象分配的效率

  2. JVM内存模型清晰版(结合代码分步解析)
    在这里插入图片描述

public class Math {

    private static final int INIT_DATA = 666;
    private static User user = new User();

    public int compute(){
        int a = 1;
        int b = 2;
        int c = (a + b) * 10;
        return c;
    }

    public static void main(String[] args) {
        Math math = new Math();
        int compute = math.compute();
        System.out.println(compute);
    }
}

拿Math类来解释执行步骤:
当执行Math类的main方法时,会在JVM栈内存中开辟一块内存区域,程序运行main方法,在新开辟的内存区域中开辟一块栈帧内存区域,并且压入栈底,此栈帧区域中包含许多小的内存区域;如:局部变量表,操作数栈,动态链接,方法出口… 通过每一行代码的执行来解析

  • 程序执行main方法中的Math math = new Math()
    先执行了Math类的无参构造方法,然后将Math类的静态常量至于JVM的方法区中,并且赋值,User对象存于JVM的堆内存中,方法区中存的是User对象在堆内存中的地址;initData存于方法区中,并且赋值666;
    main方法的栈帧内存区域中,开辟一块新的内存区域局部变量表,将math对象存于JVM堆内存中,并且将堆中的地址放到局部变量表中
  • 程序执行int compute = math.compute()
    在栈线程中新开辟一个compute方法的栈帧内存区域,采用后进先出的原理,compute()方法的栈帧区域在main栈帧区域上面;
    1. 执行int a = 1,先在栈帧区域中开辟一块内存区域局部变量表,将局部变量a放入,并赋初始值0,然后再将1赋值给a
    2. 执行int b = 2,将局部变量b放入局部变量表中,并赋初始值0,然后再将2赋值给b
    3. 执行int c = (a + b) * 10,开辟一个新的内存区域操作数栈,并且放入一个待操作的数10,并且从局部变量表中获取a和b的值放入操作数栈中,然后出栈执行运算,并将运算得到的结果30压入操作数栈中, 此时的操作数栈只有30一个数据
    4. 执行return c, main方法中执行math.compute()时,会在compute()方法栈帧内存区域中开辟一个新的内存区域方法出口,记录main方法中调用的位置,以便compute()方法执行完后,将结果返回到main方法中的代码执行位置

3. 总结

  • 当一个线程开始运行时,会在JVM的栈内存空间中开辟一个区域供线程运行使用
  • new出来的对象是存在堆内存空间中,栈中只存对象在堆中的位置地址
  • 方法区,记录线程中涉及的一些常量,静态变量类信息,其中对象常量或者静态变量存的都是对象在堆中的位置地址
  • 先进后出,如: 先执行的main方法,main的栈帧内存空间在线程栈的栈底,而main方法中调用的comput的栈帧区域在main的上部,从代码流程上看,调用的方法肯定是先执行完,所以后进入线程栈空间的comput栈帧,会先被GC销毁
  • 一个方法对应一个栈帧区域
  • 程序计数器:是用于存放下一条指令所在单元的地址的地方(记录当前线程执行到哪行代码了), 做标记
  • 本地方法栈: 一个线程调用Java方法或者本地方法时的栈
栈帧成员说明
局部变量表一组变量值存储空间,用于存放方法参数和方法内定义的局部变量,容量以变量槽(Variable Slot)为最小单位,一个槽可以存放一个32位的数据,局部变量表存放的都是变量在方法区中位置,JVM是通过索引定位的方式查找对应的变量
操作数栈也称操作栈,是一个后入先出栈(FILO),当一个方法刚刚开始执行时,其操作数栈是空的,随着方法执行和字节码指令的执行,会从局部变量表或对象实例的字段中复制常量或变量写入到操作数栈,再随着计算的进行将栈中元素出栈到局部变量表或者返回给方法调用者,也就是出栈/入栈操作。一个完整的方法执行期间往往包含多个这样出栈/入栈的过程。
动态链接在一个class文件中,一个方法要调用其他方法,需要将这些方法的符号引用转化为其在内存地址中的直接引用,而符号引用存在于方法区中的运行时常量池。Java虚拟机栈中,每个栈帧都包含一个指向运行时常量池中该栈所属方法的符号引用,持有这个引用的目的是为了支持方法调用过程中的动态连接(Dynamic Linking)。这些符号引用一部分会在类加载阶段或者第一次使用时就直接转化为直接引用,这类转化称为静态解析。
另一部分将在每次运行期间转化为直接引用,这类转化称为动态连接
方法出口有两种方式,正常完成出口,异常完成出口
①正常完成出口:指方法正常完成并退出,没有抛出任何异常,如果当前方法正常完成,则根据当前方法返回的字节码指令,这时有可能会有返回值传递给方法调用者(调用它的方法),或者无返回值。具体是否有返回值以及返回值的数据类型将根据该方法返回的字节码指令确定。
②异常完成出口:指方法执行过程中遇到异常,并且这个异常在方法体内部没有得到处理,导致方法退出。
补充说明:无论方法采用何种方式退出,在方法退出后都需要返回到方法被调用的位置,程序才能继续执行。方法退出过程实际上就等同于把当前栈帧出栈,因此退出可以执行的操作有:恢复上层方法的局部变量表和操作数栈,把返回值(如果有的话)压如调用者的操作数栈中,调整PC计数器的值以指向方法调用指令后的下一条指令。

三. 从Jvisualvm来研究下对象内存流转模型

  1. 当线程执行时,会产生大量的对象,这些对象一开始会放在堆内存空间的Eden区域中,当Eden区域达到内存峰值时,会触发Minor GC进行垃圾回收,未回收的对象被称为非垃圾对象,会移到s0区域中,并且这些对象的分代年龄增加1。
  2. 当Eden区域再次达到内存峰值时,再次触发Minor GC进行垃圾回收,此时的Minor GC不仅会扫描Eden区域,而且还会扫描s0区
  3. 当Eden区域又达到内存峰值时, Minor GC会扫描Eden区域,s0和s1区域,并且将非垃圾对象移到s0区域,对象的分代年龄加1。
  4. s0和s1区域的对象会相互转移,当对象的分代年龄达到15时,会移到老年代区域中,当老年代区域达到内存峰值时,会触发Full GC,扫描整个JVM的堆内存区域和方法区,当Full GC都无法阻止老年代区域被填满,就会报OOM(内存溢出)

演示对象在堆内存各分代区域中的流转:

public class HeapTest {

  byte[] a = new byte[1024*100];  //100kb

  public static void main(String[] args) throws InterruptedException {
      List<HeapTest> heapTests = new ArrayList<>();
      while (true){
          heapTests.add(new HeapTest());
          Thread.sleep(10);
      }
  }
}

最后代码执行结果
在这里插入图片描述
由于HeapTests对象一直是非垃圾对象,所以GC无法回收,所以Eden、S0、S1的对象在经过一次次的GC,分代年龄一直增加,直到将老年代的内存区域撑爆,最终OOM内存溢出。

执行jvisualvm命令,打开JDK自带的监控工具
在这里插入图片描述

四. GC Root与STW机制

当执行Minor GC或者Full GC时,会执行STW(停止整个事件),停止用户所有线程,GC执行完,再恢复。

  1. ★GC过程中为什么会执行STW机制?
    因为如果不执行STW,一边进行GC,一边线程继续执行,那么当线程执行完时,会进行
    出栈操作,所有在栈中开辟的内存区域全部释放,而GC无法处理的非垃圾对象此时没
    有引用,而会被GC当作垃圾对象清理,在现实的生产环境中,这显然是不允许的。

  2. JVM优化,就是减少Full GC和Minor GC的次数
    尽可能让对象都在新生代里分配和回收,尽量别让太多对象频繁进入老年代,避免频繁对老年代进行垃圾回收,同时给系统充足的内存大小,避免新生代频繁的进行垃圾回收。

五. JVM参数设置通用模型

在这里插入图片描述

  1. Spring Boot程序的JVM参数设置格式(Tomcat启动直接加在bin目录下catalina.sh文件里):
    java ‐Xms2048M ‐Xmx2048M ‐Xmn1024M ‐Xss512K ‐XX:MetaspaceSize=256M ‐XX:MaxMetaspaceSize=256M ‐jar microservice‐eurek a‐server.jar

  2. 关于元空间的JVM参数有两个:-XX:MetaspaceSize=N-XX:MaxMetaspaceSize=N
    -XX:MaxMetaspaceSize:设置元空间最大值,默认是-1,即不限制,或者说只受限于
    本地内存大小。
    -XX:MetaspaceSize:指定元空间触发Full GC的初始阈值(元空间无固定初始大小),
    以字节为单位,默认是21M,达到该值就会触发Full GC进行类型卸载

  3. 同时收集器会对该值进行调整:如果释放了大量的空间,就适当降低该值;
    如果释放了很少的空间,那么在不超过-XX:MaxMetaspaceSize(如果设置了的话)的情况下,适当提高该值。这个跟早期JDK版本的-XX:PermSize(代表永久代的初始容量)参数意思不一样,

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

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1037467.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

USB总线-Linux内核USB3.0主机控制器驱动框架分析(十二)

1.概述 如下图所示&#xff0c;Linux内核中USB主机体系结构由五部分组成&#xff0c;分别为Application Software、USB Class Driver、USB Core(USB Driver)、USB Host Controller Driver、USB Host Controller。应用程序处于用户空间&#xff0c;通过系统调用访问Class Drive…

C进阶-数据的存储

数据类型介绍 内置类型&#xff1a; //数据类型中的内置类型 // char //字符数据类型 // short //短整型 // int //整型 // long //长整型 // long long //更长的整型 // float //单精度浮点数 // double //双精度浮点数 //数据类型中的内置类型 单位是字节 // char //字…

使用 ggbreak 包进行Y轴多次截断

简介 最近在科研中需要比较不同模型的预测性能&#xff0c;我们使用相对偏差&#xff0c;均方根误差等来比较结果。此时&#xff0c;可能会得到以下结果&#xff1a; 上图出现以下问题&#xff1a; 问题一&#xff1a;由于经典模型&#xff08;Normal&#xff09;在复杂数据中…

【Unity基础】5.动画曲线

【Unity基础】5.动画曲线 大家好&#xff0c;我是Lampard~~ 欢迎来到Unity基础系列博客&#xff0c;所学知识来自B站阿发老师~感谢 &#xff08;一&#xff09;曲线编辑 (1) 动画曲线 上周我们创建了一个简单动画&#xff0c;让我们的矩形在第0帧时Y轴从0的位置&#…

ElementUI基本介绍及登录注册案例演示

目录 前言 一.简介 二.优缺点 三.Element完成登录注册 1. 环境配置及前端演示 1.1 安装Element-UI模块 1.2 安装axios和qs(发送get请求和post请求) 1.3 导入依赖 2 页面布局 2.1组件与界面 3.方法实现功能数据交互 3.1 通过方法进行页面跳转 3.2 axios发送get请求 …

008_第一代软件系统架构

第一代软件系统架构 文章目录 第一代软件系统架构项目介绍软件架构和软件构架系统框架硬件组成运行系统基础库软件层 系统架构 关键字&#xff1a; Qt、 Qml、 关键字3、 关键字4、 关键字5 项目介绍 欢迎来到我们的 QML & C 项目&#xff01;这个项目结合了 QML&…

SD-MTSP:萤火虫算法(FA)求解单仓库多旅行商问题MATLAB(可更改数据集,旅行商的数量和起点)

一、萤火虫算法&#xff08;FA&#xff09;简介 萤火虫算法(Firefly Algorithm&#xff0c;FA)是Yang等人于2009年提出的一种仿生优化算法。 参考文献&#xff1a;田梦楚, 薄煜明, 陈志敏, et al. 萤火虫算法智能优化粒子滤波[J]. 自动化学报, 2016, 42(001):89-97. 二、单仓…

Go-Python-Java-C-LeetCode高分解法-第七周合集

前言 本题解Go语言部分基于 LeetCode-Go 其他部分基于本人实践学习 个人题解GitHub连接&#xff1a;LeetCode-Go-Python-Java-C Go-Python-Java-C-LeetCode高分解法-第一周合集 Go-Python-Java-C-LeetCode高分解法-第二周合集 Go-Python-Java-C-LeetCode高分解法-第三周合集 …

力扣:105. 从前序与中序遍历序列构造二叉树(Python3)

题目&#xff1a; 给定两个整数数组 preorder 和 inorder &#xff0c;其中 preorder 是二叉树的先序遍历&#xff0c; inorder 是同一棵树的中序遍历&#xff0c;请构造二叉树并返回其根节点。 来源&#xff1a;力扣&#xff08;LeetCode&#xff09; 链接&#xff1a;力扣&am…

RASP初识

需要了解的东西. 是什么 拦截日志&#xff1a;rasp/logs/alarm/alarm.log RASP&#xff08;Runtime application self-protection&#xff09;运行时应用自我保护。 官方英译应用程序不应将大部分运行时保护委托给外部设备。应用程序应该能够自我保护&#xff08;即&#xf…

排序:希尔排序(Shell Sort)算法分析

1.算法思想 希尔排序:先追求表中元素部分有序再逐渐逼近全局有序. 希尔排序∶先将待排序表分割成若干形如 L [ i , i d , i 2 d . . . . , i k d ] L[i,i d,i 2d ...., i kd] L[i,id,i2d....,ikd]的“特殊”子表&#xff0c; 对各个子表分别进行直接插入排序。缩小增量…

1018 锤子剪刀布

一.问题&#xff1a; 大家应该都会玩“锤子剪刀布”的游戏&#xff1a;两人同时给出手势&#xff0c;胜负规则如图所示&#xff1a; 现给出两人的交锋记录&#xff0c;请统计双方的胜、平、负次数&#xff0c;并且给出双方分别出什么手势的胜算最大。 输入格式&#xff1a; …

Baumer工业相机堡盟工业相机如何通过BGAPI SDK设置相机的图像剪切(ROI)功能(C++)

Baumer工业相机堡盟工业相机如何通过BGAPI SDK设置相机的图像剪切&#xff08;ROI&#xff09;功能&#xff08;C&#xff09; Baumer工业相机Baumer工业相机的图像剪切&#xff08;ROI&#xff09;功能的技术背景CameraExplorer如何使用图像剪切&#xff08;ROI&#xff09;功…

急救车工业路由器应用提升急救效率:车联网、数据采集与远程诊疗

急救车作为医院里医疗急救过程中的重要组成部分&#xff0c;在智慧医疗物联网领域中急救车应用4G工业路由器实现网络部署与数据采集&#xff0c;通过工业4G路由器能够实时采集到病患的生理数据、救护现场音频与视频、GPS定位以及车辆运行状态等重要信息。这些数据将被传输到医疗…

【产品资料】产品经理面试问题(一)

今天和大家免费分享产品经理常见的面试题目&#xff0c;含回答思路分析和回答事例。 更多的产品文档、原型模板、视频教程等资源&#xff0c;请关注公众号&#xff1a;Axure高保真原型

6、SpringBoot_项目的打包与运行

七、SpringBoot项目的打包与运行 1.目前项目怎么运行的 通过浏览器访问idea 将jar部署到服务器 2.maven 打包项目 命令 mvn package使用命令后会得到如下的jar 3.程序运行 命令 java -jar 项目.jar启动如下 4.springboot打包需要插件 插件 <plugin><group…

梯度下降法

梯度下降法 对于一个二元一次函数 y ax b&#xff0c;我们只需要知道两个 (x&#xff0c;y) 点即可获取到 a、b 的值&#xff0c;我们称其为精确解&#xff0c;如下图&#xff1a; 但是如果该函数中存在已知分布的噪声&#xff0c;那么又该如何求解&#xff1a; 我们可以假…

如何修复wmvcore.dll缺失问题,wmvcore.dll下载修复方法分享

近年来&#xff0c;电脑使用的普及率越来越高&#xff0c;人们在日常生活中离不开电脑。然而&#xff0c;有时候我们可能会遇到一些问题&#xff0c;其中之一就是wmvcore.dll缺失的问题。wmvcore.dll是Windows平台上用于支持Windows Media Player的动态链接库文件&#xff0c;如…