【从零开始学习JVM | 第五篇】快速了解运行时数据区

news2025/1/23 2:08:16

前言:

        当谈论 Java 程序的运行机制时,JVM(Java 虚拟机)的运行时数据区是一个必不可少的话题。JVM 运行时数据区是 Java 程序在运行过程中分配内存和管理数据的重要区域,它包括了方法区、堆、虚拟机栈、程序计数器和本地方法栈等部分。了解 JVM 运行时数据区不仅有助于理解 Java 程序的运行原理,还有助于优化程序的性能和解决一些与内存管理相关的问题。本文将从各个方面介绍 JVM 运行时数据区的结构和作用,希望能够为读者提供全面深入的了解。

目录

前言:

运行时数据区:

程序计数器:

Java虚拟机栈:

本地方法栈:

堆: 

方法区: 

杂项知识点:

总结:


 

 

运行时数据区:


 

接下来我详细的讲解一下每一部分的作用:

程序计数器:

程序计数器(Program Counter Register)是 JVM(Java 虚拟机)中的一块较小的内存区域,它是线程私有的,即每个线程都有自己的程序计数器。程序计数器的作用是指示当前线程正在执行的虚拟机字节码指令的地址或者下一条需要执行的指令地址

以下是程序计数器的一些重要作用:

  1. 指示下一条指令的地址:程序计数器存储着当前线程正在执行的虚拟机指令的地址,当线程被调度并恢复执行时,JVM 将根据程序计数器中的地址来获取下一条需要执行的指令。

  2. 线程切换时保持状态:由于程序计数器是线程私有的,因此在线程切换时,程序计数器中的值会被保存和恢复。这保证了线程在恢复执行时能够继续执行之前的指令,而不会出现混乱。

  3. 支持线程中断和恢复:程序计数器的状态可以用来支持线程的中断和恢复机制。当一个线程被中断或者阻塞后又恢复执行时,程序计数器可以确保线程能够从中断前的地方继续执行,而不会跳转到其他位置。

  4. 处理异常跳转:程序计数器还用于处理异常跳转,例如在发生异常时,程序计数器可以指示 JVM 跳转到异常处理代码的指令地址。

在 Java 虚拟机规范中,程序计数器被定义为 JVM 中的一部分,并且针对程序计数器的操作都是 JVM 指令集中的一部分。程序计数器的大小是固定的,且不会发生内存溢出或内存泄露的情况,因为它不涉及到对象的分配或垃圾回收

Java虚拟机栈:

Java 虚拟机栈(JVM Stack)是 Java 虚拟机(JVM)中的一块重要内存区域,用于存储方法的局部变量、操作数栈、动态链接、返回地址以及方法出口等信息。每个线程在创建时都会在虚拟机栈中分配一个栈帧(Stack Frame),每当线程调用一个方法时,JVM 都会在虚拟机栈中创建一个对应的栈帧,用于存储该方法的相关信息

所谓的栈帧,就是一个保存方法基本信息的容器。

以下是JAVA虚拟机栈的一些重要特点和作用:

  1. 线程私有的数据区域:与方法区和堆不同,Java 虚拟机栈是线程私有的数据区域,意味着每个线程在运行时都拥有自己的虚拟机栈,用于存储线程独享的方法调用信息。

  2. 栈帧的结构:每个栈帧包含局部变量表(Local Variable Table)操作数栈(Operand Stack)动态链接(Dynamic Linking)方法返回地址额外的附加信息。局部变量表用于存储方法的参数和局部变量,操作数栈用于存储方法执行过程中的操作数,动态链接用于指向当前方法在运行时常量池中的方法引用,而方法返回地址用于存储方法调用后的返回地址。
  3. 栈帧的入栈与出栈:当线程执行方法调用时,对应的栈帧被压入虚拟机栈,当方法执行结束后,该栈帧被弹出栈。这种入栈与出栈的操作是基于方法的调用和返回关系进行的。

  4. 支持方法的递归调用:虚拟机栈的存在支持了方法的递归调用,每次递归调用都会在虚拟机栈中创建一个新的栈帧,以便存储方法的局部变量和执行信息。

  5. 栈深度限制:JVM 使用虚拟机栈来管理方法的调用和返回,因此虚拟机栈的深度是有限制的。如果方法调用的层次太深,虚拟机栈会发生栈溢出(StackOverflowError)。

Java 虚拟机栈在程序执行期间起着至关重要的作用,它不仅存储方法的局部变量和执行信息,还支持了方法的调用和返回。而JAVA虚拟机栈如果栈帧过多,占用内存超过栈内存可以分配的最大大小就会出现内存溢出。

本地方法栈:

本地方法栈(Native Method Stack)是 Java 虚拟机(JVM)中的一块内存区域,用于支持执行 Java 虚拟机调用本地(Native)方法时的数据结构。与虚拟机栈类似,本地方法栈也是线程私有的,每个线程都有自己的本地方法栈,用于执行本地方法时的方法调用和返回。

以下是本地方法栈的一些重要特点和作用:

  1. 支持本地方法调用:本地方法栈可以理解为虚拟机栈用于执行本地方法的部分。当 Java 虚拟机调用本地方法时,本地方法栈会记录该调用的信息,包括参数、局部变量等。

  2. 与虚拟机栈的区别:虚拟机栈主要用于执行 Java 方法时的数据结构,而本地方法栈用于执行本地方法时的数据结构。两者在结构上存在类似之处,但在功能上有明显的区别。

  3. 本地方法栈的深度限制:类似于虚拟机栈,本地方法栈也有一定的深度限制。在执行本地方法调用时,如果本地方法栈的深度超出了限制,则会导致栈溢出错误。

  4. 安全性和性能:本地方法栈的存在主要是为了提高 Java 虚拟机与本地方法库的安全性和性能,使得 Java 虚拟机能够与本地代码进行无缝集成。

需要注意的是,和虚拟机栈一样,本地方法栈也属于 Java 虚拟机规范中定义的一部分,不同的虚拟机对本地方法栈的大小和结构可能略有差异,但其作用和功能是相似的。

总体来说,本地方法栈是 Java 虚拟机用于支持本地方法调用的重要内存区域,通过了解本地方法栈的结构和作用,可以更好地理解 Java 虚拟机与本地方法库的交互过程,提高代码的运行效率和安全性。

在Hotspot虚拟机中,Java虚拟机栈和本地方法栈实现上使用了同一个栈空间


堆: 

是用于动态内存分配的一块内存区域,用来存储对象实例和数组。在 Java 中,所有的对象实例和数组都在堆上分配内存。堆内存支持动态分配和释放,通过堆上的内存分配指针进行分配。这意味着对象可以在程序运行时动态创建,并且可以根据需要动态调整堆的大小。堆内存在 Java 虚拟机中受到垃圾回收器的管理,用于回收不再使用的对象,释放其占用的内存。Java 的垃圾回收机制主要针对堆内存进行,以确保内存的合理利用和程序的稳定性。

堆的结构划分:

  1. 新生代(Young Generation):用于存放新创建的对象。通常被划分为 Eden 区和两个 Survivor 区。大部分对象在这里被创建,然后经过几轮垃圾回收后如果仍然存活,则会被移到老年代。

  2. 老年代(Old Generation):用于存放存活时间较久的对象,即由新生代经过多次垃圾回收后依然存活下来的对象。

  3. 永久代/元空间(PermGen/Metaspace):在较早的 Java 版本中使用永久代(PermGen)来存放类的元数据、常量池等信息,但在较新的版本中,使用元空间(Metaspace)来代替。这部分内存主要用于存放类和方法的元信息,以及一些静态的数据。

堆内存的划分:

  1. Used(已使用:表示当前已经被使用的堆内存大小,即已经被分配给对象实例和数组的内存空间的大小。

  2. Total(总量):表示当前堆内存的总大小,即JVM当前所分配的堆内存的总量。包括已使用的内存和未被使用的内存。

  3. Max(最大值):表示堆内存的最大可用空间大小,即JVM所能申请到的最大堆内存大小。

而在实际业务中,我们会直接把Total设置为和Max一样的大小,这样避免了申请并分配内存时间上的开销。同时也不会出现内存过剩后堆收缩的情况。堆也是可以溢出的。

方法区: 

方法区(Method Area)是Java虚拟机(JVM)的一个重要组成部分,它用于存储已被加载的类信息、常量、静态变量、即时编译器编译后的代码等数据。在虚拟机规范中,方法区是线程共享的。

方法区在虚拟机启动时被创建,并且是一块连续的内存区域。它的大小可以固定,也可以根据需要进行动态扩展。在HotSpot虚拟机中,方法区的大小是固定的,可以通过设置JVM参数来调整。

方法区主要存储以下内容:

  1. 类信息:包括类的完整结构、字段、方法、继承关系、接口等。

  2. 运行时常量池:每个类都有一个运行时常量池,在方法区中进行存储。它包含了字面量和符号引用。在早期,字符串常量池是运行时常量池的一部分,而在后续将二者进行了拆分

  3. 静态变量:所有类的静态变量都存放在方法区中。

  4. 即时编译器编译后的代码:JVM会将热点代码进行即时编译,并将生成的本地机器码存储在方法区中。

以下是方法区的一些重要特点和作用:

  1. 存储元数据:方法区主要用于存储类的元数据、常量、静态变量以及类中的符号引用等信息。这些信息在类加载时被存储在方法区中,对所有实例对象都是共享的。

  2. 无需手动内存管理:方法区不需要像堆内存(heap)一样进行垃圾回收。这是因为方法区中存储的数据并不像堆内存中的对象一样动态地创建和销毁,而是在类加载时确定并且通常在程序运行过程中保持不变。

  3. 永久代到元空间的变迁:在较老的JVM版本中,方法区通常被实现为永久代(PermGen)。但是从Java 8开始,永久代被元空间(Metaspace)所取代。元空间不再受到默认的最大永久代大小的限制,而是根据系统内存动态扩展。

  4. 动态性:与永久代不同,元空间的大小并不受默认设置或者-Xmx参数的限制,它可以根据应用的实际需要动态变化。

  5. 类信息存储:方法区将类的结构信息、即时编译的代码、常量池、静态变量等存储在内存中,这些数据对于程序执行过程中的类加载、方法调用等起着关键作用。

虽然历代设计中方法区的空间一直很大,但是他仍然有内存溢出的风险。

杂项知识点:

1.一个字符对象如何判断是存储到字符串常量池还是堆中?

在Java中,字符串对象有可能存储在堆内存中,也有可能存储在字符串常量池中,这取决于字符串对象创建的方式。

  1. 字符串常量池: 当使用字面量形式创建字符串对象时,例如 String s = "Hello",这个字符串对象会被存储在字符串常量池中。字符串常量池是Java堆内存中的一部分,用于存储字面量形式创建的字符串对象,这样的设计可以避免重复存储相同内容的字符串。

  2. 堆内存: 当使用 new 关键字显式创建字符串对象时,例如 String s = new String("Hello"),这个字符串对象会被存储在堆内存中。这种方式会在堆内存中创建一个新的字符串对象,即使字符串内容在常量池中已经存在。

因此,字符串对象既可以存储在堆内存中,也可以存储在字符串常量池中,取决于字符串对象的创建方式。需要注意的是,通过 intern() 方法可以将堆内存中的字符串对象手动放入字符串常量池中

总结:

JVM(Java虚拟机)运行时数据区是Java程序执行过程中存储和管理数据的内存区域,它被划分为多个不同的区域,包括方法区、堆、虚拟机栈、本地方法栈和程序计数器等。

首先,方法区(在Java 8及之前称为永久代)存储每个类的结构信息、静态变量、常量以及编译后的方法字节码。它在运行时可以被多个线程共享,是被所有线程共享的内存区域之一。

其次,堆是存储对象实例和数组的内存区域,它是Java程序中最常用的数据结构之一。堆的特点是可以动态分配内存,当程序运行时可以动态地创建对象实例,而且它是所有线程共享的内存区域。

虚拟机栈用于存储线程执行方法的局部变量、操作数栈、动态链接、方法出口等信息。每个方法执行的同时都会创建一个栈帧用于存储局部变量和操作数,而栈帧则会随着方法执行的结束而被销毁。

本地方法栈则与虚拟机栈类似,不同在于虚拟机栈是为Java方法服务,而本地方法栈则是为native方法(使用C、C++等语言编写的方法)服务。

最后,程序计数器是当前线程所执行的字节码的行号指示器,它在多线程环境下为每个线程都分配一个独立的程序计数器,用于记录当前线程执行的位置,是线程私有的内存区域。

总的来说,JVM运行时数据区域在Java程序执行过程中起着至关重要的作用,通过合理管理这些数据区域,可以优化程序的性能和内存的利用,也有助于理解Java程序的执行机制和内存管理原理

如果我的内容对你有帮助,请点赞,评论,收藏。创作不易,大家的支持就是我坚持下去的动力!

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

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

相关文章

TA-Lib学习研究笔记(九)——Pattern Recognition (4)

TA-Lib学习研究笔记(九)——Pattern Recognition (4) 最全面的形态识别的函数的应用,通过使用A股实际的数据,验证形态识别函数,用K线显示出现标志的形态走势,由于入口参数基本上是o…

js/jQuery常见操作 之 jQuery操作复选框的常见问题

js/jQuery常见操作 之 jQuery操作复选框的常见问题 1. js/jQuery的其他一些常见基础操作2. 全选/全不选问题2.1 效果2.2 实现代码2.2.1 简单js实现2.2.2 jQuery实现2.2.2.1 注意语法(区别jQuery版本)2.2.2.2 完整代码实现 3. jQuery实现点击 行tr 实现ch…

【GAMES101】三维变换

games101的第四节课讲了三维变换和观察变换,我们这里先记录一下三维变换的知识,后面再讲观察变换 齐次坐标下的三维变换 类似于解决之前二维变换平移的问题,三维变换下用齐次坐标通过增加一个维度来表示,第四个维度为1表示这是个…

自动数据增广论文笔记 | AutoAugment: Learning Augmentation Strategies from Data

谷歌大脑出品 paper: https://arxiv.org/abs/1805.09501 这里是个论文的阅读心得,笔记,不等同论文全部内容 文章目录 一、摘要1.1 翻译1.2 笔记 二、(第3部分)自动增强:直接在感兴趣的数据集上搜索最佳增强策略2.1 翻译2.2 笔记 三、跳出论文&#xff0c…

Python超级详细的raise用法

当程序出现错误时,系统会自动引发异常。除此之外,Python 也允许程序自行引发异常,自行引发异常使用 raise 语句来完成。 很多时候,系统是否要引发异常,可能需要根据应用的业务需求来决定,如果程序中的数据…

智能配电房解决方案

智能配电房解决方案运用物联网、云计算、大数据、人工智能等先进技术,构建具备遥测、遥信、遥控、遥调、遥视功能的智能配电房,依托电易云-智慧电力物联网实现对配电室的全面感知、智能监控、优化运行、精益管理,提高供电可靠性,降…

系统设计之数据库

为您的项目选择正确的数据库是一项复杂的任务。许多数据库选项都适合不同的用例,很快就会导致决策疲劳。 我们希望这份备忘单提供高级指导,以找到符合您项目需求的正确服务并避免潜在的陷阱。 注意:Google 关于其数据库用例的文档有限。尽管…

笔记本用gpu运行tensorflow-gpu,keras写的老程序,结果与原来不一样,一脸懵逼。

先说结论我笔记一是rtx3050ti, 重点RTX30系列最低要求CUDA版本为11.1,否则最后跑程序会报错。再说现象,突发奇想想在笔记本上运行一个以前在1080titensorflow-gpu1.5.2,keras2.2.4上面写的一个图像分类模型,先用cpu模式 运行一下一切正常。如…

探索正则可视化工具:让编程更直观、高效

导语:在当今的编程世界中,正则表达式已成为不可或缺的技能。然而,理解和编写正则表达式往往是一项具有挑战性的任务。为了降低门槛,提高编程效率,正则可视化工具应运而生。 一、正则表达式的简介与历史 正则表达式&a…

基于JNI实现调用C++ SDK

基于JNI实现调用C SDK 背景分析解决实践 背景 上篇文章总结了几种Java项目调用C/C SDK项目方法,在逐一实践、踩坑后,最终还是敲定采用 JNI 方式进行实现。在文章开始的过程,会先大概讲讲笔者遇到的情况,因为封装方式需要根据实际…

文本润色工具有哪些,高质量的文本润色软件

在当今信息过载的时代,文本的重要性愈发凸显。即便是最精心构思的文章,若未经过仔细的润色,也难以达到最佳的表达效果。本文将专心分享文本润色工具的种类。 文本润色工具的种类 文本润色工具根据其功能和应用范围可以分为多个种类&#xff…

【C语言快速学习基础篇】之二控制语句、循环语句、隐式转换

文章目录 一、控制语句1.1、for循环1.2、while循环1.3、注意:for循环和while循环使用上面等同1.4、do while循环1.4.1while条件成立时1.4.2、while条件不成立时 C语言介绍 C语言是一门面向过程的计算机编程语言,与C、C#、Java等面向对象编程语言有所不同…

python pydoc生成API文档

pydoc是python内置的一个文档生成模块。 pydoc 模块会根据 Python 模块来自动生成文档。 生成的文档可在控制台中显示为文本页面,提供给 Web 浏览器访问或者保存为 HTML 文件。 对于模块、类、函数和方法,显示的文档内容取自文档字符串(即 _…

202350读书笔记|《再别康桥:徐志摩诗选》——微风起,清芬酝藉,不减荼

202350读书笔记|《再别康桥:徐志摩诗选》——微风起,清芬酝藉,不减荼 《再别康桥:徐志摩诗选》我觉得有时候诗人是很狂热的,上头的感觉。 有几首很喜欢,节选如下: 偶然 我是天空里的一片云&…

【STM32】TIM定时器输入捕获

1 输入捕获 1.1 输入捕获简介 IC(Input Capture)输入捕获 输入捕获模式下,当通道输入引脚出现指定电平跳变时(上升沿/下降沿),当前CNT的值将被锁存到CCR中(把CNT的值读出来,写入到…

JavaScript编程基础 – 闭包(Closure)

JavaScript编程基础 – 闭包 JavaScript Programming Essentials - Closure By JacksonML 闭包和JavaScript的作用域有关。 我们需要先理解闭包的概念。 本文简要介绍闭包函数以及环境状态,并用实例说明闭包的创建及其基本用法。希望对学习及开发有所帮助。 1. …

sql优化常用的几种方法

SQL优化是数据库管理和应用开发中的一个重要环节,它涉及到对SQL查询语句的改进,以提高执行效率,减少系统负载,并优化资源使用。以下是一些常用的SQL优化方法,我会尽量覆盖一些关键的点。 使用合适的数据类型 选择适…

通俗易懂的案例+代码解释AOP 切面编程

目录 1. 理解AOP2 Before2.1 controller层2.2 service层2.3 自定义注解2.4 切面 advice 3 After4 Around spring的三大核心:IOC控制反转、DI依赖注入、AOP面向切面编程 刚开始接触springboot项目,前两个使用的多,亲自使用AOP的机会并不多&…

python自动化测试如何做数据缓存 ?这个第三方包推荐给你,方便又简单!

1.数据缓存说明 数据缓存可以说也是项目开发中比不可少的一个工具 ,像我们测试的系统中 ,你都会见到像Redis一样的数据缓存库 。使用缓存数据库的好处不言而喻,那就是效率高 ,简单数据直接放在缓存中 ,存取简单方便 。…