【Java】虚拟机(JVM)内存模型全解析

目录

一、运行时数据区域划分

版本的差异:

二、程序计数器

程序计数器主要作用

三、Java虚拟机

1. 虚拟机运行原理

2. 活动栈被弹出的方式

3. 虚拟机栈可能产生的错误

4. 虚拟机栈的大小

四、本地方法栈

五、堆

1. 堆区的组成:新生代+老生代

2. 堆空间的大小设置

3. 创建对象的内存分配

4. 堆区产生的错误

六、字符串常量

1. String 的两种创建方式

2. String 的intern()方法:

3. String的拼接

4. String s1 = new string("abc");这句代码创建了几个字符串对象?


        

        JVM 虚拟机在执行 Java 程序的过程中,会把它管理的内存划分成若干个不同的区域,每个区域有各自的不同的用途、创建方式及管理方式。有些区域随着虚拟机的启动一直存在,有些区域则随着用户线程的启动和结束而建立和销毁,这些共同组成了 Java 虚拟机的运行时数据区域,也被称为JVM内存模型

一、运行时数据区域划分

        JVM 虚拟机在执行 Java 程序的过程中会把它管理的内存划分成若干个不同的数据区域。由方法区堆区虚拟机栈本地方法栈程序计数器五部分组成。

        

版本的差异:

        JDK 1.8 之分为:线程共享( Heap 堆区、Method Area 方法区)、线程私有(虚拟机栈、本地方法栈、程序计数器)。

        JDK 1.8 以分为:线程共享( Heap 堆区、MetaSpace 元空间)、线程私有(虚拟机栈、本地方法栈、程序计数器)。       

                                                               ​​​​​​        

        其中虚拟机栈、本地方法栈、程序计数器是线程私有的区域,所以随着线程消亡而结束。而线程共享的 Heap 堆区、 Metaspace 元空间会随着虚拟机的启动,一直存在

二、程序计数器

        程序计数器是一块较小的内存空间,是当前线程所执行的字节码的行号指示器。

程序计数器主要作用

  • 字节码解释器通过改变程序计数器来依次读取指令,从而实现代码的流程控制,如:顺序执行、选择、循环、异常处理。
  • 在多线程的情况下,程序计数器用于记录当前线程执行的位置,从而当线程被切换回来的时候,能够知道当前线程的运行位置,恢复当前线程的执行。

        程序计数器是唯一一个不会出现 OutofMemoryError 的内存区域,它随着线程的创建而创建,随着线程的结束而死亡。

三、Java虚拟机

        虚拟机栈是线程执行 Java 程序时,处理 Java 方法中内容的内存区域。虚拟机栈也是线程私有的区域,每个 Java方法被调用的时候,都会在虚拟机栈中创建出一个栈帧,而每个栈帧又由局部变量表操作数栈动态链接方法返回四部分组成,有些虚拟机的栈帧还包括了一些附加信息。

        JVM 内存区域可以粗略的区分为堆内存(Heap)和栈内存(stack)。其中栈就是 JVM Stack 虚拟机栈,或者说是虚拟机栈中局部变量表部分,局部变量表主要存放了编译期可知的各种基本数据类型变量值( boolea、byte、char、short、int、float、long、double )、对象引用(reference 类型,它不同于对象本身,可能是一个指向对象起始地址的引用指针,也可能是指向一个代表对象的句柄或其他与此对象相关的位置)。

1. 虚拟机运行原理

        每一次方法调用都会有一个对应的栈帧被压入 JVM Stack 虚拟机栈,每一个方法调用结束后,代表该方法的栈帧会从 JVM Stack 虚拟机栈中弹出。虚拟机栈是线程的私有区域,并且栈帧不允许被其他线程访问,所以不存在线程安全问题,栈帧弹出后就内存就会被系统回收,所以不也存在垃圾回收问题。

        在活动线程中,只有位于栈顶的帧才是有效的,称为当前活动栈帧,代表正在执行的当前方法。在 JVM 执行引擎运行时,所有指令都只能针对当前活动栈帧进行操作。虚拟机栈通过 pop 和 push 的方式,对每个方法对应的活动栈帧进行运算处理,方法正常执行结束,肯定会跳转到另一个栈帧上。

2. 活动栈被弹出的方式

        Java 方法有两种返回方式,不管哪种返回方式都会导致当前活动栈帧被弹出:

  • return 语句
  • 抛出异常

3. 虚拟机栈可能产生的错误

        Java 虚拟机栈会出现两种错误:StackOverFlowErrorOutOfMemoryError

  • StackOverFlowError:当线程请求栈的深度超过 JVM 虚拟机栈的最大深度的时候,就抛出 StackOverFlowError 错误。
  • OutofMemoryError:JVM 的内存大小可以动态扩展,如果虚拟机在动态扩展栈时无法申请到足够的内存空间,则抛出 OutOfMemoryError 异常。

4. 虚拟机栈的大小

        虚拟栈的大小可以通过 -Xss 参数设置,默认单位是 byte,也可以使用k,m,g作为单位(不区分大小写)。例如:-xss 1m

        在不同操作系统下的 -xss 默认值不同:

  •         Linux:1024k
  •         MacOs:1024k
  •         Windows:默认值依赖于虚拟机的内存

四、本地方法栈

        native 关键字修饰的本地方法被执行的时候,在本地方法栈中也会创建一个栈帧,用于存放该 native 本地方法的局部变量表、操作数栈、动态链接、方法出口信息。方法执行完毕后,相应的栈帧也会出栈并释放内存空间。也会出现StackOverFlowErrorOutOfMemoryError 两种错误。

五、堆

        Heap 堆区,用于存放对象实例数组的内存区域。

        Heap 堆区,是 JVM所管理的内存中最大的一块区域,被所有线程共享的块内存区域。堆区中存放对象实例,“几乎”所有的对象实例以及数组都在这里分配内存。

1. 堆区的组成:新生代+老生代

        从垃圾回收的角度,由于现在收集器基本都采用分代垃圾收集思想,所以JVM 中的堆区往往进行分代划分,例如:新生代和老年代。目的是更好地回收内存,或者更快地分配内存

        堆区的组成分为新生代(Young Generation)、老年代(Old Generation)。新生代被分为伊甸区(Eden)和幸存者区(from+to),幸存区又被分为Survivor 0(from)和Survivor 1(to)。

        新生代和老年代比例为1:2,伊甸区和s0、S1比例为 8:1:1,不同区域存放对象的用途和方式不同:
        
        (1)伊甸区( Eden):存放大部分新创建对象。

        (2)幸存区(Survivor):存放 Minor Gc 之后,Eden 区和幸存区( Survivor)本身没有被回收的对象。
        
        (3)老年代:存放 Minor Gc 之后且年龄计数器达到 15 依然存活的对象、 Major GC 和 Full GC 之后仍然存活的对象。

        

2. 堆空间的大小设置

        堆区的内存大小是可修改的,默认情况下,初始堆内存为物理内存的 1/64,最大为物理内存的 1/4。

  • -xms:设置初始堆内存,例如:-Xms64m
  • -Xmx:设置最大堆内存,例如:-Xmx64m
  • -Xmn:设置新生代内存,例如:-Xmx32m

3. 创建对象的内存分配

        创建一个新对象,在堆中的分配内存。

        大部分情况下,对象会在Eden区生成,当den 区装填满的时候,会触发 Young Garbage collection,即 YGC 垃圾回收的时候,在 Eden 区实现清除策略,没有被引用的对象则直接回收。
依然存活的对象会被移送到 Survivor区。Survivor 区分为s0和s1 两块内存区域。每次 Gc 的时候,它们将存活的对象复制到未使用的 Survivor空间(s0 或 s1),然后将当前正在使用的空间完全清除,交换两块空间的使用状态。每次交换时,对象的年龄会加 +1。
        
        如果 YGC 要移送的对象大于 Survivor 区容量的上限,则直接移交给老年一个对象也不可能永远呆在新生代,在中 一个对象从新生代晋升到老年代的阈值默认值是 15,可以在 Survivor 区交换 14 次之后,晋升至老年代。

        

4. 堆区产生的错误

        堆区最容易出现的就是 OutOfMemoryError 错误,这种错误的表现形式会
有以下两种:

        (1)OutOfMemoryErrorGc overhead Limit Exceeded:当JVM 花太多
时间执行垃圾回收,并且只能回收很少的堆空间时,就会发生此错误。
        
        (2)OutofMemoryErrorJava heap space:假如在创建新的对象时,堆内存
中的空间不足以存放新创建的对象,就会引发此错误。

六、字符串常量

1. String 的两种创建方式

  • 第一种方式是在常量池中获取字符串对象
  • 第二种方式是直接在堆内存空间创建一个新的字符串对象

2. String 的intern()方法:

        检查指定字符串在常量池中是否存在?如果存在,则返回地址,如果不存在,则在常量池中创建
        

3. String的拼接

        String str1 = "str";String str2 = "ing";String str3 ="str"+"ing"; // 常量池中的新字符串对象String str4= str1 + str2; //在堆中创建的新字符串对象String str5 ="string"; // 常量池中的已有字符串对象System.out.println(str3 == str4); //falseSystem.out.println(str3 == str5); //trueSystem.out.println(str4 == str5); //false

4. String s1 = new string("abc");这句代码创建了几个字符串对象?

        创建 1或 2 个字符串。如果常量池中已存在字符串常量“abc ”,则只会在堆空间创建一个字符串常量“ abc ”。如果常量池中没有字符串常量“ abc”,那么它将首先在池中创建,然后在堆空间中创建,因此将创建总共2个字符串对象。

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

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

相关文章

速通LLaMA3:《The Llama 3 Herd of Models》全文解读

文章目录 概览论文开篇IntroductionGeneral OverviewPre-TrainingPre-Training DataModel ArchitectureInfrastructure, Scaling, and EfficiencyTraining Recipe Post-TrainingResultsVision ExperimentsSpeech Experiments⭐Related WorkConclusionLlama 3 模型中的数学原理1…

目标检测系列(一)什么是目标检测

目录 一、相关名词解释 二、目标检测算法 三、目标检测模型 四、目标检测应用 五、目标检测数据集 六、目标检测常用标注工具 一、相关名词解释 关于图像识别的计算机视觉四大类任务: 分类(Classification):解决“是什么&…

【LLM开源项目】LLMs-微调框架-LLaMA-Factory入门指南v3.0

【#】SFT 训练 项目地址:https://llamafactory.readthedocs.io/zh-cn/latest/getting_started/sft.html 可以使用以下命令进行微调: llamafactory-cli train examples/train_lora/llama3_lora_sft.yamlexamples/train_lora/llama3_lora_sft.yaml 提供…

车辆识别数据集,图片数量20500,模型已训练200轮

车辆识别数据集(Vehicle Recognition Dataset, VDRD) 摘要 VDRD 是一个专为车辆识别设计的大规模数据集,它包含了20500张不同类型的汽车、货车、公交车以及其他类型车辆的图像。数据集提供了四种车辆类别:汽车、货车、其他车辆和…

单片机原理及应用

单片机原理 一、单片机概述1.1 各种单片机系列1.2嵌入式处理器1.2.1 嵌入式DSP1.2.2 嵌入式微处理器 二、AT89S522.1 硬件组成2.2 引脚功能2.2.1 电源、时钟引脚2.2.2 控制引脚2.2.3 并行I/O口引脚 2.3 AT89S52单片机的CPU2.3.1 运算器2.3.2 控制器 2.4 AT89S52单片机的存储器结…

【代码随想录训练营第42期 Day60打卡 - 图论Part10 - Bellman_ford算法系列运用

目录 一、Bellman_ford算法的应用 二、题目与题解 题目一:卡码网 94. 城市间货物运输 I 题目链接 题解:队列优化Bellman-Ford算法(SPFA) 题目二:卡码网 95. 城市间货物运输 II 题目链接 题解: 队列优…

MySQL_表_进阶(1/2)

我们的进阶篇中,还是借四张表,来学习接下来最后关于表的需求,以此完成对表的基本学习。 照例给出四张表: 学院表:(testdb.dept) 课程表:(testdb.course) 选课表:(testdb.sc) 学生表…

python调用c++动态链接库,环境是VS2022和vscode2023

目录 前言:配置环境:基础夯实(对于ctypes的介绍):1. 加载共享库2. 定义函数原型3. 调用函数4. 处理数据结构5. 处理指针6. 错误处理7. 使用 ctypes.util总结 效果展示:操作步骤(保姆级教学)一在VS中创建dll…

YOLOv8——测量高速公路上汽车的速度

引言 在人工神经网络和计算机视觉领域,目标识别和跟踪是非常重要的技术,它们可以应用于无数的项目中,其中许多可能不是很明显,比如使用这些算法来测量距离或对象的速度。 测量汽车速度基本步骤如下: 视频采集&#x…

(done) 声音信号处理基础知识(4) (Understanding Audio Signals for ML)

来源:https://www.youtube.com/watch?vdaB9naGBVv4 模拟信号特点如下 时域连续(x轴) 振幅连续(y轴) 如下是模拟信号的一个例子: 数字信号特点如下: 一个离散值序列 数据点的值域是一系列有限的值 ADC:模拟信号到数字信号的…

python全栈学习记录(十八)re、os和sys、subprocess

re、os和sys、subprocess 文章目录 re、os和sys、subprocess一、re1.正则字符2.正则表达式的使用3.group的使用4.贪婪匹配与惰性匹配5.其他注意事项 二、os和sys1.os2.sys 三、subprocess四、打印进度条 一、re python中的re模块用来使用正则表达式,正则就是用一系…

【Python机器学习】NLP信息提取——提取人物/事物关系

目录 词性标注 实体名称标准化 实体关系标准化和提取 单词模式 文本分割 断句 断句的方式 使用正则表达式进行断句 词性标注 词性(POS)标注可以使用语言模型来完成,这个语言模型包含词及其所有可能词性组成的字典。然后,该…

http增删改查四种请求方式操纵数据库

注意:在manage.py项目入口文件中的路由配置里,返回响应的 return语句后面的代码不会执行,所以路由配置中每个模块代码要想都执行,不能出现return 激活虚拟环境:venv(我的虚拟环境名称)\Scripts\activate …

java项目发布后到Tomcat时,总是带一层路径解决方案

java项目发布后到Tomcat时,总是带一层路径 参考文章:java 线上项目访问项目 会多一层项目根路径 根据参考文章写的这篇文章,部分文章细节有完善和改动 在Java Web应用中,当你把应用发布到Tomcat时,如果应用的web.xml配置文件中的&…

Karmada新版本发布,支持联邦应用跨集群滚动升级

摘要:本次升级支持联邦应用跨集群滚动升级,使用户版本发布流程更加灵活可控;透明同事karmadactl 新增了多项运维能力,提供独特的多集群运维体验。 本文分享自华为云社区 《Karmada v1.11 版本发布!新增应用跨集群滚动升…

柔性数组 初学版

1.定义 结构中的最后⼀个元素允许是未知⼤⼩的数组,这就叫做『柔性数组』成员 有些编译器会报错⽆法编译可以改成: typedef struct st_type { int i; int a[]; // 柔性数组成员 }type_a; 2.柔性数组的特点: • 结构中的柔性数组成员前…

ReadWriteLock读写锁

读写锁基本概念 ReadWriteLock是Java并发包中的一个接口,它定义了两种锁:读锁(Read Lock)和写锁(Write Lock),真正的实现类是ReentrantReadWriteLock。读锁允许多个线程同时读取共享资源&#…

JAVA开源项目 体育馆管理系统 计算机毕业设计

本文项目编号 T 048 ,文末自助获取源码 \color{red}{T048,文末自助获取源码} T048,文末自助获取源码 目录 一、系统介绍二、演示录屏三、启动教程四、功能截图五、文案资料5.1 选题背景5.2 国内外研究现状5.3 可行性分析5.4 用例设计 六、核…

记一次Mac 匪夷所思终端常用网络命令恢复记录

一天莫名奇妙发现ping dig 等基础命令都无法正常使用。还好能浏览器能正常访问&#xff0c;&#xff0c;&#xff0c;&#xff0c; 赶紧拿baidu试试^-^ ; <<>> DiG 9.10.6 <<>> baidu.com ;; global options: cmd ;; connection timed out; no serve…

美业门店怎么提升业绩?连锁美业门店管理系统收银系统拓客系统源码

美业门店想要提升业绩&#xff0c;需要考虑多方面的因素&#xff0c;并采取综合性的方法。以下是一些可以考虑的因素和建议&#xff1a; 产品与服务优化&#xff1a; 提供高质量的美容产品和服务&#xff0c;确保顾客满意度。不断更新产品线&#xff0c;引入新的时尚趋势&#…