【JVM】(一)深入理解JVM运行时数据区
文章目录
- 一、JVM 运行流程
- 二、虚拟机栈(线程私有)
- 三、本地方法栈 (线程私有)
- 四、方法区(元数据区)
- 五、堆(线程共享)
- 六、程序计数器(线程私有)
一、JVM 运行流程
JVM 是 Java 程序的运行基础和运行环境,同时也是 Java 实现 "一次编译,到处运行"
的关键所在。因此,深入了解 JVM 对于学习和理解 Java 编程语言是至关重要的,那么JVM 到底是如何运行的呢?
下面这张图片展示了 JVM 的基本运行过程:
JVM的执行过程涉及以下主要组成部分::
-
Java 代码转换为字节码(
.class
文件):编写完成后的 Java 源代码需要通过 Java 编译器编译为字节码,生成.class
文件,这些文件包含了 Java 程序的中间代码。 -
类加载器(ClassLoader):JVM 的类加载器负责将
.class
字节码文件加载到内存中的运行时数据区
。类加载器根据类的全限定名(Fully Qualified Name)查找并加载对应的字节码文件,并根据文件内容创建 Class 对象来代表这个类,以供后续的执行引擎调用。 -
运行内存管理:JVM负责管理程序运行时的内存区域,主要包括以下几个区域:
- 虚拟机栈(Java Virtual Machine Stacks):用于存储方法调用的栈帧,包含局部变量、操作数栈等。
- 本地方法栈(Native Method Stacks):用于执行本地方法的栈。
- 方法区(Method Area):存储类的结构信息、常量池、静态变量等。
- 堆(Heap):存储对象实例和数组的内存区域。
- 程序计数器(Program Counter):记录当前线程执行的字节码指令的地址或索引。
-
执行引擎(Execution Engine):执行引擎是 JVM 的核心组件之一,负责执行加载到内存中的字节码文件。执行引擎有两种方式执行字节码:
- 解释执行:逐条解释字节码指令并执行相应的操作。解释执行效率较低,但跨平台性好,适用于刚开始执行的代码段或是执行次数较少的代码段。
- 编译执行:将字节码编译成特定平台的本地代码,然后交由CPU执行。编译执行效率高,但需要额外的编译时间,适用于执行次数频繁的代码段。
-
本地方法库(Native Libraries):JVM 中的 Java 代码无法直接访问底层操作系统,因为字节码只是一套跨平台的指令集规范。当 Java 代码需要执行底层操作系统或与本地代码(如C、C++)进行交互的时候。就需要通过本地方法库来实现这些功能。本地方法库允许 Java 程序调用与操作系统相关的本地代码,从而实现整个程序的功能。
以上就是 JVM 的主要运行过程以及运行时涉及的组成部分,下文是对其中的运行时数据区的详细介绍。
二、虚拟机栈(线程私有)
虚拟机栈是 Java 线程私有的内存区域,用于存储线程的方法调用和局部变量等信息。虚拟机栈的生命周期和线程相同,虚拟机栈描述的是 Java 方法执行的内存模型:
- 每个方法在执行的时候都会创建一个栈帧(Stack Frame)用于存储局部变量表、操作数栈、动态链接、方法出口等信息。
- 当方法结束的时候,响应的栈帧也会因为出栈而销毁。
虚拟机栈主要包含四个部分:
- 局部变量表:用于存储方法的局部变量。在Java方法执行时,它会分配一块内存区域,用于存储方法参数和方法内部定义的局部变量。局部变量表的大小在编译期间就被确定,存储的数据类型包括基本数据类型和对象引用。
- 操作栈:用于执行方法时的计算操作。Java虚拟机的字节码指令通常都是基于操作数栈进行运算的。当一个方法被调用时,会将参数值和返回值等压入操作数栈,在方法执行时,字节码指令会从操作数栈中取值进行计算。
- 动态链接:用于指向运行时常量池中该栈帧所属方法的引用。在Java虚拟机的运行时常量池中,存放着每个类的方法的符号引用,动态链接将这些符号引用与实际内存地址进行关联。
- 方法返回地址:用于指示方法的返回地址。当方法执行完成后,需要知道从哪里继续执行,方法返回地址就是用来记录返回的目标地址。
什么是线程私有:
- 线程私有是指在多线程环境下,每个线程都拥有自己独立的内存区域,其他线程无法直接访问或修改该区域。线程私有内存中的数据对于每个线程都是独立的,互不影响。这种设计使得多线程程序可以同时执行,每个线程都能够独立地运行和维护自己的数据。
- 在JVM中,
虚拟机栈
、本地方法栈
和程序计数器
是线程私有的内存区域。每个线程都有自己的虚拟机栈和本地方法栈,用于支持方法调用和执行,并且这些栈的数据对其他线程是不可见的。程序计数器也是线程私有的,每个线程都有自己的程序计数器,用于指示当前线程执行的字节码指令的地址或索引。线程的切换时,会保存和恢复程序计数器的值,以保证线程切换后能正确继续执行。- 其他共享内存区域如
堆
和方法区(元空间)
是线程共享的。堆用于存储Java对象实例和数组,方法区用于存储类的结构信息、常量池、静态变量等。多个线程可以共同访问堆和方法区的数据,因此在多线程环境下,需要通过同步机制来保护共享数据的一致性和正确性。- 内存区域按线程私有和共享的划分:
三、本地方法栈 (线程私有)
本地方法栈与虚拟机栈类似,也是 Java 线程私有的内存区域。它用于支持 Java 程序调用和执行本地方法(Native Method)。本地方法是使用其他语言(如C、C++)编写的方法,通过本地方法接口(JNI,Java Native Interface)与 Java 代码进行交互。本地方法栈和虚拟机栈在功能上也是类似的,但它们分别用于 Java 方法和本地方法的调用。
四、方法区(元数据区)
方法区是 Java 线程共享的内存区域,用于存储类的结构信息、常量池、静态变量、即时编译器编译后的代码等。在 JDK 8 及以前版本,方法区是永久代(Permanent Generation)
;而在 JDK 8 及以后版本,方法区被替换为元空间(Metaspace)
。方法区的大小可以通过启动JVM 时的参数来设置。
在《Java虚拟机规范中》把此区域称之为 “方法区”,而在 HotSpot 虚拟机的实现中,在 JDK 7 时此区域叫做永久代(Permanent Generation),JDK 8 中叫做元空间(Metaspace)。
JDK 1.8 元空间的变化:
- 对于 HotSpot 来说,JDK 8 元空间的内存属于本地内存,这样元空间的大小就不在受 JVM 最大内存的参数影响了,而是与本地内存的大小有关。
- JDK 8 中将字符串常量池移动到了堆中。
运行时常量池:
运行时常量池(Runtime Constant Pool)是方法区的一部分,用于存放在编译期间生成的各种字面量和符号引用。它是在类加载过程中,由虚拟机根据字节码文件中的常量池表构建而成。
在运行时常量池中,主要包含两种类型的数据:
-
字面量:
- 字符串字面量:即Java程序中直接写的字符串值,例如:“Hello, Java”。在JDK 8中,字符串字面量被移动到堆中,这样它们也可以被垃圾收集器回收,避免了一些内存问题。
final
常量:在编译期间可以确定的final
常量,例如:final int MAX_VALUE = 100;
。- 基本数据类型的值:例如:整数、浮点数、字符等基本数据类型的字面量。
-
符号引用:
符号引用是一种在编译期间产生的、但在类加载阶段需要用到的一种数据结构,它用于描述被引用的目标。符号引用包括:- 类和接口的完全限定名:例如:
java.lang.String
。 - 字段的名称和描述符:用于描述字段的名称和数据类型,例如:
int count
。 - 方法的名称和描述符:用于描述方法的名称和参数列表以及返回值类型,例如:
void print(String message)
。
- 类和接口的完全限定名:例如:
运行时常量池在类加载后,会存放在方法区中,并且在整个类的生命周期中存在,与类本身一起被回收。它在程序运行时提供了常量池解析、动态链接、方法调用等功能,为 Java 程序的执行提供了必要的支持。
五、堆(线程共享)
堆(Heap)是 Java 虚拟机中的一个运行时数据区域,用于存储Java对象实例和数组。堆是JVM中最大的一块内存区域,也是唯一被所有线程共享的内存区域。
在Java程序运行过程中,当使用new
关键字创建对象时,对象实例会被分配在堆中。同时,数组也是对象,因此数组的元素也会存储在堆中。堆的大小可以在启动 JVM 时通过参数来指定,也可以动态调整(如果未指定大小,则JVM会根据系统内存自动设置初始大小)。
堆的主要特点包括:
-
线程共享:堆是所有线程共享的内存区域。所有线程都可以访问堆中的对象实例,这使得多个线程可以共同操作和共享对象。
-
动态分配和回收:堆的大小在程序运行时是可以动态调整的。当创建对象时,JVM会自动分配堆中的内存空间。当对象不再被引用时,垃圾收集器会自动回收堆中不再使用的对象的内存。
-
垃圾回收:堆中的内存由垃圾收集器负责管理。垃圾收集器会周期性地检查堆中的对象,将不再被引用的对象标记为垃圾,然后回收这些垃圾对象的内存,释放给堆供其他对象使用。
-
自动内存管理:Java 中的堆内存由 JVM 自动进行内存管理,程序员不需要手动释放内存。JVM 会自动进行垃圾回收,释放不再使用的内存,避免了内存泄漏等问题。
另外,在Java虚拟机的堆内存中,通常被划分为两个主要区域:新生代(Young Generation)
和老生代(Old Generation)
。
-
新生代(Young Generation):
新生代是存放新创建的对象的区域。在新生代中,通常会将堆内存划分为一个Eden空间和两个Survivor空间(通常称为S0和S1)。新创建的对象首先会被分配到Eden空间。当Eden空间满时,会触发Minor GC(Young GC),垃圾回收器会将Eden空间和其中还存活的对象复制到一个未使用的Survivor空间中。然后,垃圾回收器会清除Eden空间和正在使用的Survivor空间,将其中的垃圾对象回收。幸存下来的对象会被晋升到老生代。 -
老生代(Old Generation):
老生代是存放长时间存活的对象的区域。老生代中存放的对象通常是在新生代经过一定次数的Minor GC后仍然存活的对象,或者是大对象等。当老生代空间满时,会触发Major GC(Full GC),垃圾回收器会对整个堆进行回收,包括新生代和老生代的所有对象。
通过将堆内存分为新生代和老生代,并采用不同的垃圾回收策略,可以提高垃圾回收的效率。新生代中的Minor GC频繁进行,回收生命周期短的对象,尽可能快速地释放内存;而老生代中的Major GC则相对较少进行,回收生命周期长的对象,保证了老生代的稳定性和可靠性。
六、程序计数器(线程私有)
程序计数器也是 Java 线程私有的内存区域。它是一种指示器,用于指示当前线程执行的字节码指令的地址或索引
。在Java线程切换时,程序计数器的值会被保存和恢复,以保证线程切换后能正确继续执行。
相关文章:

【JVM】(一)深入理解JVM运行时数据区
文章目录 一、JVM 运行流程二、虚拟机栈(线程私有)三、本地方法栈 (线程私有)四、方法区(元数据区)五、堆(线程共享)六、程序计数器(线程私有) 一、JVM 运行流…...

C++ QRegExpValidator
//正在表达式限制输入 QString str "\\b(?:(?:25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)\\.){3}(?:25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)\\b"; ui->lineEdit->setValidator(new QRegExpValidator(QRegExp(str))); //用于占位 ui->lineEdit->setI…...

备战秋招 | 笔试强训19
目录 一、选择题 二、编程题 三、选择题题解 四、编程题题解 一、选择题 1、二分查找的时间复杂度() A. O(N*log(N)) B. O(N) C. O(log(N)) D. O(N^2) 2、有一个单向链表中有一个A、B两个相邻元素,有一个指针p指向元素A,现将…...

第一章 计算机网络概述
第一章 计算机网络概述 1.1 计算机网络在信息时代的作用 1.2 因特网概述 网络分类: 网络:许多计算机连接在一起的的局域网; 互联网:internet许多网络连接在一起; 因特网:Internet 全球最大的互联网&…...

谷粒商城第六天-商品服务之分类管理下的获取三级分类树形列表
目录 一、总述 1.1 前端思路 1.2 后端思路 二、前端部分 2.1 在网页中建好目录及菜单 2.1.1 建好商品目录 2.1.2 建好分类管理菜单 编辑 2.2 编写组件 2.2.1 先完成组件文件的创建 2.2.2 编写组件 2.2.2.1 显示三级分类树形列表 三、后端部分 3.1 编写商品分类…...

【UI自动化测试】Appium+Python+Unittest+HTMLRunner
简介 获取AppPackage和AppActivity 定位UI控件的工具 脚本结构 PageObject分层管理 HTMLTestRunner生成测试报告 启动appium server服务 以python文件模式执行脚本生成测试报告 【B站最通俗易懂】Python接口自动化测试从入门到精通,超详细的进阶教程,看完…...

【限时优惠】红帽openstack管理课程(CL210) 即将开课
课程介绍 通过实验室操作练习,学员将能够深入学习红帽企业 Linux OpenStack 平台各服务的手动安装方法,还将了解 OpenStack 开发社区的未来发展计划。 培训地点: 线下面授:苏州市姑苏区干将东路666号401室; 远程…...

Golang之路---02 基础语法——函数
函数 函数定义 func function_name( [parameter list] ) [return_types] {函数体 }参数解释: func:函数由 func 开始声明function_name:函数名称,函数名和参数列表一起构成了函数签名。[parameter list]:参数列表&a…...

数据结构和算法入门(时间/空间复杂度介绍--java版)
数据结构和算法入门(时间/空间复杂度介绍–java版) write in front 作者: 向大佬学习 专栏: 数据结构(java版) 作者简介:大二学生 希望能学习其同学和大佬的经验! 本篇博客简介&…...

Spring Mvc 文件上传(MultipartFile )—官方原版
一、创建应用程序类 要启动Spring Boot MVC应用程序,首先需要一个启动器。在这个示例中,已经添加了spring-boot-starter thymelaf和spring-boot-starter web作为依赖项。要使用Servlet容器上传文件,您需要注册一个MultipartConfigElement类&…...

【E题】2023年电赛运动目标控制与自动追踪系统方案
系统的设计和制作可以按照以下步骤进行: 设计红色光斑位置控制系统: 选择合适的红色激光笔,并将其固定在一个二维电控云台上。 使用电机和编码器来控制电控云台的水平和垂直运动。 设计一个控制电路,可以通过输入控制信号来控制…...

企业网络安全之零信任和身份认证
零信任并不是一种技术,而是一个安全概念,是一种建立安全战略的理念、方法和框架。 零信任提供了一系列概念和思想,其中心思想是怀疑一切,否定一切,不再以网络边界为限,不能再将内部网络定义为可信任的&…...

【雕爷学编程】MicroPython动手做(28)——物联网之Yeelight 5
知识点:什么是掌控板? 掌控板是一块普及STEAM创客教育、人工智能教育、机器人编程教育的开源智能硬件。它集成ESP-32高性能双核芯片,支持WiFi和蓝牙双模通信,可作为物联网节点,实现物联网应用。同时掌控板上集成了OLED…...

[运维|中间件] 东方通TongWeb使用笔记
参考文献 东方通tongweb部署服务 东方通tongweb部署服务 使用笔记 默认访问地址 http://ip:9060/console/默认用户名密码 TongWeb7.0默认用户名密码:thanos,thanos123.com...

WIZnet W6100-EVB-Pico DHCP 配置教程(三)
前言 在上一章节中我们讲了网络信息配置,那些网络信息的配置都是用户手动的去配置的,为了能跟电脑处于同一网段,且电脑能成功ping通板子,我们不仅要注意子网掩码,对于IP地址主机位和网络位的划分,而且还要注…...

【Linux】Ansible 脚本 playbook 剧本
提示:文章写完后,目录可以自动生成,如何生成可参考右边的帮助文档 Ansible 脚本 playbook 剧本 playbook 剧本Templates 模块tags 模块Roles 模块在一个 playbook 中使用 roles 的步骤 playbook 剧本 playbooks 本身由以下各部分组成 &#…...

解决 tensorflow 出现的 ImportError: Could not find the DLL(s) ‘msvcp140_1.dll‘. 问题
在安装完tensorflow库后出现 问题详述: ImportError: Could not find the DLL(s) msvcp140_1.dll. TensorFlow requires that these DLLs be installed in a directory that is named in your %PATH% environment variable. You may install these DLLs by downlo…...

百度与AI:历史、投资和监管
来源:猛兽财经 作者:猛兽财经 百度的人工智能在中国具有先发优势 随着ChatGPT的爆火,人工智能重新引起了投资者的注意,然而人工智能并不是突然爆火的,而是全球众多公司在人工智能技术上进行数十年如一日的研发和积累&a…...

Kafka3.0.0版本——Broker(Zookeeper服务端存储的Kafka相关信息)
目录 一、启动zookeeper集群及kafka集群服务启动1.1、先启动三台zookeeper集群服务,再启动三台kafka集群服务1.2、使用PrettyZoo连接zookeeper客户端工具 二、在zookeeper服务端存储的Kafka相关信息 一、启动zookeeper集群及kafka集群服务启动 1.1、先启动三台zook…...

【图论】无向图连通性(tarjan算法)
割边:dfn[u]<low[v] 割点:dfn[u]<low[v] (若为根节点,要有两个v这样的点) 一.知识点: 1.连通: 在图论中,连通性是指一个无向图中的任意两个顶点之间存在路径。如果对于图中的任意两个顶点 u 和 v&…...

Docker安装
Docker实践 yum安装 YUM源可以使用官方YUM源、清华大学开源镜像站配置YUM源,也可以使用阿里云开源镜像站提供的YUM源,建议选择使用阿里云开源镜像站提供的YUM源,原因速度快。 地址: https://developer.aliyun.com/mirror/ 我们安装ce版 …...

06. 计数原理
6. 计数原理 6.1 分类加法计数原理与分步乘法计数原理 分类加法计数原理定义 完成一件事,有 n n n 类办法,在第1类办法中有 m 1 m_1 m1 种不同的方法,在第2类办法中有 m 2 m_2 m2 种不同的方法,…,在第 n n…...

计算机网络基础(静态路由,动态路由,公网IP,私网IP,NAT技术)
文章目录 一:静态路由和动态路由二:静态路由的配置路由信息的方式演示三:默认路由四:公网IP和私网IP和NAT技术的基本理解 一:静态路由和动态路由 在说静态路由和动态路由前,我们需要来了解一下࿰…...

CGAL 点云Alpha-Shape曲面重建算法
文章目录 一、简介二、相关参数三、实现代码四、实现效果参考资料一、简介 在数学上, a l p h a − s h a p e alpha-shape a...

Java 文件过滤器FileFilter | 按条件筛选文件
文章目录 一、概述1.1 何时会用到文件过滤器1.2 工作流程1.3 常用的接口和类1.4 文件过滤器的作用 二、按文件属性过滤2.1 按前缀或后缀过滤文件名2.2 按文件大小过滤 三、按文件内容过滤3.1 文本文件过滤器3.1.1 根据关键字过滤文件内容3.1.2 使用正则表达式过滤文件内容 3.2 …...

python格式化地址信息
背景 最近在折腾一个好玩的库,capa 实现地址的格式化输出。我看的教程是这样的: location_str ["徐汇区虹漕路461号58号楼5楼", "泉州市洛江区万安塘西工业区"] import cpca df cpca.transform(location_str) df在正式的运行代码…...

k8s1.26.6 安装gitlab
Gitlab官方提供了 Helm 的方式在 Kubernetes 集群中来快速安装,但是在使用的过程中发现 Helm 提供的 Chart 包中有很多其他额外的配置,所以我们这里使用自定义的方式来安装,也就是自己来定义一些资源清单文件。 Gitlab主要涉及到3个应用&…...

C5.0决策树建立个人信用风险评估模型
通过构建自动化的信用评分模型,以在线方式进行即时的信贷审批能够为银行节约很多人工成本。本案例,我们将使用C5.0决策树算法建立一个简单的个人信用风险评估模型。 导入类库 读取数据 #创建编码所用的数据字典 col_dicts{} #要编码的属性集 cols [che…...

【k8s集群部署】使用containerd运行时部署kubernetes集群(V1.27版本)
【k8s集群部署】使用containerd运行时部署kubernetes集群(V1.27版本) 一、本次实践介绍1.1 环境规划介绍1.2 本次实践简介二、三台主机基础环境配置2.1 主机配置工作2.2 关闭防火墙和selinux2.3 关闭swap2.4 清空iptables2.5 配置时间同步2.6 修改内核参数2.7 配置hosts文件三…...

网络安全进阶学习第八课——信息收集
文章目录 一、什么是信息收集?二、信息收集的原则三、信息收集的分类1.主动信息收集2.被动信息收集 四、资产探测1、Whois查询#常用网站: 2、备案信息查询#常用网站: 3、DNS查询#常用网站: 4、子域名收集#常用网站:#常…...