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

目录

      • 一. 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];  //100kbpublic 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.mzph.cn/news/86833.shtml

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

相关文章

改写软件-怎么选择改写软件

什么是改写软件&#xff1f;改写软件是基于自然语言处理技术的工具&#xff0c;它们可以分析一段文字&#xff0c;并将其重新表达&#xff0c;以保持原始意义&#xff0c;但使用不同的词汇和结构。这种技术可用于减少内容的重复&#xff0c;增加多样性&#xff0c;或者简化复杂…

Python量化交易学习笔记(0)

本文将简单回顾我的量化交易学习的历程&#xff0c;并给出新手学习量化交易的建议学习路线&#xff0c;适合于尚无稳定盈利策略的量化新手阅读&#xff0c;量化大神们请略过。 本文将在博客中置顶&#xff0c;并不定期根据我的学习、交易进行更新。 回顾学习历程 2020年初接…

使用爬虫批量下载图片链接并去重

设置timeout(20, 20), verifyFalse避免超时和校验问题jpeg以jpg格式保存获取图片编码的md5并存为文件名&#xff0c;以避免重复 import pandas as pd import requests import os import hashlib from tqdm import tqdm file_path xiaofang.xlsx save_dir xiaofang df pd.re…

SpringMVC初级

文章目录 一、SpringMVC 概述二、springMVC步骤1、新建maven的web项目2、导入maven依赖3、创建controller4、创建spring-mvc.xml配置文件&#xff08;本质就是spring的配置件&#xff09;5、web.xml中配置前端控制器6、新建a.jsp文件7、配置tomcat8、启动测试 三、工作流程分析…

C# ComboBox 和 枚举类型(Enum)相互关联

C# ComboBox 和 枚举类型(Enum)相互关联 目的 在C# Winform面板上的ComboBox选择项&#xff0c;由程序填写某个Enum的各个枚举项目。 在运行中读取ComboBox的选择项&#xff0c;返回Enum数值。 非编程方法 低阶做法可以在winform设计窗口手动填写&#xff0c;但是不会自动跟…

pytorch环境搭建到pycharm项目映射配置(成功后回顾性记录/自用)

利用Anaconda创建pytorch虚拟环境 前提&#xff1a;成功安装Anaconda&#xff0c;确保可以打开NVIDIA控制面板 开始-》搜索“Anaconda Prompt” 打开后输入&#xff1a;conda create -n 你的虚拟环境名 python3.9。输入y&#xff0c;继续安装&#xff0c;完成。 输入&#…

合肥综合性国家科学中心人工智能研究院-机器学习作业(一)

1.试析min-max规范化和z-score规范化的优缺点 可参考博客&#xff1a;https://wenku.csdn.net/answer/fdbf30eb204644e5b69fc533a3757268 2.试分析损失函数与性能度量的关系 损失函数和性能度量之间的关系可以根据优化目标来理解。损失函数的优化目标是最小化预测值与实际值之…

C语言 cortex-A7核 点LED灯 (附 汇编实现、使用C语言 循环实现、使用C语言 封装函数实现【重要、常用】)

1 汇编实现 text global _start start: ************** LED1点灯 ---> PE10 **************/ ************** RCC章节初始化 **************/ CC_INIT:1.使能GPIOE组控制器&#xff0c;通过RCC_MP_AHB4ENSETR寄存器设置GPIOE组使能0x50000A28[4] 1ldr r0,0x50000A28 准…

力扣刷题-链表-两两交换链表中的节点

24.两两交换链表中的节点 给定一个链表&#xff0c;两两交换其中相邻的节点&#xff0c;并返回交换后的链表。你不能只是单纯的改变节点内部的值&#xff0c;而是需要实际的进行节点交换。 解题思路 采用正常模拟的方法。 建议使用虚拟头结点&#xff0c;这样会方便很多&am…

算法训练第六十五天|螺旋遍历二维数组

LCR 146. 螺旋遍历二维数组 - 力扣&#xff08;LeetCode&#xff09; 总结&#xff1a;本质是模拟一个螺旋的过程&#xff0c;其中关键是如何限制边界条件或者说是循环结束条件。题目要求是按从左到右、从上到下、从右到左、从下到上的顺序&#xff0c;所以可以设置循环来完成…

面向面试知识-Redis

面向面试知识-Redis 什么是Redis 运行于内存的基于key-value的非关系型数据库。 一款开源的内存数据结构存储&#xff0c;用作数据库、缓存、消息代理等。&#xff08;可以基于Redis实现分布式锁、以及消息队列&#xff09; 发布订阅&#xff1f;&#xff1f; 对数据类型的操…

[Firefox/快捷键] 禁用Ctrl-W快捷键

最近给Firefox这一快捷键坑了几次。恰好发现在CMU也有人遇到类似的烦恼&#xff0c;找到一篇基于Linux的教程。 我使用的是Windows&#xff0c;所以根据自己的情况做了些修改&#xff0c;成功了。小众需求就犯懒直接将笔记贴上了&#xff0c;如果有谁用得上的话&#xff0c;我写…

消息队列中,如何保证消息的顺序性?

本文选自&#xff1a;advanced-java 作者&#xff1a;yanglbme 问&#xff1a;如何保证消息的顺序性&#xff1f; 面试官心理分析 其实这个也是用 MQ 的时候必问的话题&#xff0c;第一看看你了不了解顺序这个事儿&#xff1f;第二看看你有没有办法保证消息是有顺序的&#xf…

聊聊jedis连接池参数配置

序 本文主要研究一下jedis连接池的参数配置 JedisConfig redis/clients/jedis/JedisPoolConfig.java public class JedisPoolConfig extends GenericObjectPoolConfig<Jedis> {public JedisPoolConfig() {// defaults to make your life with connection pool easier…

STL map,插入和查找的一些注意事项

01、前言&#xff08;废话&#xff09; C 的 std::map 容器中插入键值对主要有myMap(std::make_pair(key value)) &#xff0c;它们的区别你了解吗&#xff1f; auto it myMap,find(key) 和 auto value myMap[key] 都可以用于在 C 的 std::map 容器中查找键对应的值&#xff…

Spring Boot的新篇章:探索2.0版的创新功能

文章目录 引言1. Spring Boot 2.0的响应式编程2. 自动配置的改进3. Spring Boot 2.0的嵌入式Web服务器4. Spring Boot 2.0的Actuator端点5. Spring Boot 2.0的Spring Data改进6. Spring Boot 2.0的安全性增强7. Spring Boot 2.0的监控和追踪8. Spring Boot 2.0的测试改进结论 &…

6、SpringBoot_项目的打包与运行

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

c++源码编译过程(翻译阶段)的若干细节概要

c程序的编译主要包含两个阶段&#xff1a;源码编译(翻译阶段)和目标文件链接。 源码编译过程主要有如下这些阶段&#xff1a; 阶段1: 翻译源码文本字符 阶段2: 逻辑源码行标准化处理 阶段3: 文法处理&#xff0c;分解为不同的源码文本类型序列。例如分解为注释、预处理指…

从零学习开发一个RISC-V操作系统(二)丨GCC编译器和ELF格式

本篇文章的内容 一、GCC&#xff08;GUN Compiler Collection&#xff09;1.1 GCC的命令格式1.2 GCC的主要执行步骤1.3 GCC涉及的文件类型 二、ELF简介2.1 ELF文件格式图2.2 ELF文件处理的相关工具2.3 练习 本系列是博主参考B站课程学习开发一个RISC-V的操作系统的学习笔记&…

机器学习第十四课--神经网络

总结起来&#xff0c;对于深度学习的发展跟以下几点是离不开的: 大量的数据(大数据)计算资源(如GPU)训练方法(如预训练) 很多时候&#xff0c;我们也可以认为真正让深度学习爆发起来的是数据和算力&#xff0c;这并不是没道理的。 由于神经网络是深度学习的基础&#xff0c;学…