当前位置: 首页 > article >正文

JVM学习专题(二)内存模型深度剖析

目录

1.JVM结构体系

​编辑

2.跨平台特性 

3.JVM整体结构及内存模型

1.栈内存

1、栈帧:

1.局部变量表

2.操作数栈

3.动态链接

4.方法出口

2、创建对象

2.程序计数器:

3.方法区

​4.堆

5.本地方法区

6.总结


1.JVM结构体系

JDKJRE JVM的包含关系:

1)JDK=JRE+ 开发工具集(例如Javac,java编译工具等)

2) JRE=JVM+JavaSE标准类库(java核心类库)

3) 如果只想运行开发好的 .class文件 只需要JRE

2.跨平台特性 

.java文件经过javac指令变成.class字节码文件 ,再通过java命令进入到java虚拟机里面运行,同样的文件到不同环境的jvm运行都会产生不同的二进制机器码,字节码是统一的,但JVM生成的机器码会因环境而异

一次编译,到处运行

每个不同版本的JDK内部都有对应的不同操作系统的jvm环境,也就是不同版本的jvm去实现的

3.JVM整体结构及内存模型

总共是有3块部分,运行流程也是如下:

1.类加载子系统

2.运行时数据区(内存模型)

3.字节码执行引擎

1.栈内存

我们来看一个简单的代码

比如我们有一个main主方法,当我们运行的时候会有一个主线程来运行这个方法,此时java虚拟机会在线程栈内分配一块独立的空间,用来存放我们线程执行过程中用到的局部变量。不同的线程执行都有自己的内存空间去放局部变量,这就是栈内存。每一个方法的局部变量都有在栈内存里面的一块栈帧内存区域来存放,每个栈帧区域都是独立的不会嵌套,这个栈就是数据结构的那个先进后出的栈,因此代码的从外到内执行变成了从上到下执行,符合!

上图代码的jvm处理字节码文件的指令:

1、栈帧:

栈帧的内部也有很多区域,如下图:

1.局部变量表

一开始存的就是具体方法的局部变量a之类的

​数据类型​​存储方式​​示例​
​基本类型​直接存储值(int, float, boolean等)int a = 10;
​对象引用​存储指向堆内存对象的指针(reference)String s = "hello";

后续会被操作数栈赋值 

2.操作数栈

比如上述代码的int a=2,常量2先通过JVM内的指令iconst_1压入操作数栈里,局部变量表中分配的一块内存空间给变量a,然后通过指令istore_1使常量2先出栈,再存入局部变量表使a=2赋值,注意,在JVM中字节码指令的执行是原子性的,istore_1​原子操作​,先弹出栈顶值,再存入局部变量表,保证每条指令在执行时 ​​不会被线程调度打断。

虽然运算过程发生在​​操作数栈​​内,但JVM执行算术指令时,​​必须先将操作数从栈顶弹出​​,运算完成后再将结果压回栈顶

3.动态链接

在运行时确定方法的实际调用地址​,比如动态链接确定实际调用的compute()方法,我们调用这个方法的时候得去知道这个方法内部有哪些指令,因为compute()已经放入常量池里面了,相当于目前只是个符号,当运行到这个符号的时候需要去解析,加载时会解析所有方法的符号引用,但​​非静态方法的绑定推迟到运行时​​(因多态,将符号引用Math.compute:()存入​​运行时常量池​​,但​​不解析具体地址​​,因可能有子类覆盖),所以compute()只能在程序运行的时候加载,程序运行的时候把符号引用转换成对应代码的内存直接地址(或者说是直接引用)​

4.方法出口

compute()方法执行完要出computr()方法的栈帧回到main()方法的栈帧里面

2、创建对象

 

当我们new了一个对象之后,这个math对象会存入堆里面,但是此时栈里面的main()方法的栈帧里面也有一个局部变量表里的math变量,这两个math的关系是:

栈:本质是一个​​引用​​,存储堆里math对象的地址,类似于指针

堆:包含对象头(类型指针、GC标记等)、实例数据(字段)和方法表(vtable)等实际内容

因此我们可以得到一个结论:栈里面的很多局部变量都是指向堆里面的地址

====================================================================

2.程序计数器:

程序计数器也是在一个线程里面的,和栈内存一样,是线程私有​​的内存区域。

作用:

1.​记录当前线程正在执行的字节码指令的地址​

2.存储下一条要执行的指令地址​

        当JVM执行字节码时,程序计数器(PC寄存器)会​​指向当前线程正在执行的指令的地址​​

3.控制程序执行流程​

        顺序执行​​:​​字节码执行引擎​​会动态修改程序计数器的值

4.线程切换后恢复执行​

        当线程被操作系统挂起(如时间片用完),PC会保存当前执行位置。

        线程恢复时,JVM根据程序计数器的值继续执行​​,确保程序逻辑正确。

====================================================================

3.方法区

方法区存的是常量池,所以也叫运行时常量池

方法区=常量+静态变量+类信息

当我们new了一个静态变量的user对象,这个对象会存入堆里面,此时user变量是存入方法区里面的,因为他是静态变量,所以这里也是方法区里面的user变量指向堆里面的user对象

因此我们又可以得到一个结论:方法区里面的很多静态变量都是指向堆里面的地址

拓展一下常量池类型:

  1. 类文件常量池​​ (Class File Constant Pool)

    • 存储在.class文件中
    • 包含编译期确定的各种符号引用和字面量
  2. ​​运行时常量池​​ (Runtime Constant Pool)

    • 每个类/接口独有的
    • 在类加载时从类文件常量池创建
  3. ​​字符串常量池​​ (String Constant Pool)

    • 专门存储字符串字面量
    • Java 7开始从方法区移到堆内存
  4. ​​基本类型包装类常量池​​

    • 如IntegerCache、LongCache等
    • 缓存特定范围内的基本类型包装对象
  5. ​​符号引用常量池​​ (Symbol Table)

    • JVM内部使用的符号表
    • 存储类、方法、字段等的符号引用
  6. ​​动态常量池​​ (Dynamic Constant Pool)

    • Java 11引入
    • 支持动态语言特性
  7. ​​本地方法常量池​​ (Native Method Constant Pool)

    • 为本地方法调用服务的常量池
  8. ​​匿名类常量池​​ (Anonymous Class Constant Pool)

    • 为匿名类特化的常量池结构

====================================================================

​4.堆

结构图:

区域分为:

堆=年轻代(Eden+s0+s1)+老年代        s0+s1:Survivor区 

我们new出来的对象大部分都放在Eden区

1、如果Eden放满了怎么办?那么字节码执行引擎开启垃圾收集线程(垃圾回收GC),会把无用的对象回收

所有GC Roots​​共同作为起点,比如静态变量区和方法区的对象开始找引用对象,当找到某个对象不再被GC Roots直接或间接引用,就是说没有任何引用链连接到GC Roots,此时这条线上的所有节点都会被标记为非垃圾对象,因此会把这些对象从Eden复制到Survivor区s0,剩下的就是垃圾对象会被删除。

2、当Eren第2次满了,这时候会再次触发上述流程(放入Eden里对象),只不过满了之后回收的区域变成了Eren+s0,非垃圾对象会从Eren+s0区到s1区,剩下的垃圾对象再次被删除。

3、如果第3次满了,再次触发上述流程,只不过回收的区域变成了Eren+s1,非垃圾对象会从Eren+s1区到s0区,剩下的垃圾对象再次被删除。

4、每挪一次,对象的分段年龄会+1,一般达到15次会进入老年区;当放入s0或者s1的时候放不下也会直接放入老年区

注意:​​静态变量区属于GC Roots​​它本身不会被回收,而是通过它判断堆内对象是否存活。通常会进入到老年代的有:静态变量、对象池、缓存对象、spring容器里的对象

====================================================================

5.本地方法区

比如start()方法里面会调用一个本地方法接口是用C++写的,通过native关键字声明的方法:

本地方法会去找.dll文件

====================================================================

6.总结

当你读完读明白整篇文章的时候,你应该就理解如下图片了:

相关文章:

JVM学习专题(二)内存模型深度剖析

目录 1.JVM结构体系 ​编辑 2.跨平台特性 3.JVM整体结构及内存模型 1.栈内存 1、栈帧: 1.局部变量表 2.操作数栈 3.动态链接 4.方法出口 2、创建对象 2.程序计数器: 3.方法区 ​4.堆 5.本地方法区 6.总结 1.JVM结构体系 JDK、JRE 和 JVM…...

密码学实验:凯撒密码

密码学实验:凯撒密码 一、实验目的 掌握凯撒密码的数学原理:理解字符移位与模运算的结合,实现加解密算法。理解暴力破解本质:通过穷举有限密钥空间,掌握利用语言特征破解密文的方法。编程实践:用Python实…...

linux备份与同步工具rsync

版权声明:原创作品,请勿转载! 文章目录 版权声明:原创作品,请勿转载! 实验环境介绍: 1.工具介绍 2.详细介绍 2.1 本地模式(用得少) 2.2 远程模式 2.3 守护进程模式…...

PYTHON训练营DAY26

一、函数 (一)不带参数的函数 # 定义一个简单的问候函数 def greet():"""打印一句问候语。"""message "大家好!欢迎学习Python函数定义!"print(message)greet()(二&#x…...

WPS一旦打开,就会修改默认打开方式,怎么解?

目录 前言 解决方法 结语 前言 电脑上同时存在WPS和微软的Office全家桶,但是我更喜欢用Office全家桶。前几天刚在设置改过来,忘记更改pdf文件打开默认应用。结果没过几天,不小心用WPS打开pdf文件时候,给我把默认设置全改回去了…...

从概念到可工程化智能体的转变路径——以“知识奇点工程师”为例

产品部门定义了一个如下概念性的“知识奇点工程师”,他们构建的不仅仅是一个数据库或知识图谱,而是一个活的、能自我进化的知识生态系统,是整个“Neuralink for Education”宏伟蓝图的基石。他们的工作难度和重要性,不亚于为AI引擎…...

单片机-STM32部分:12、I2C

飞书文档https://x509p6c8to.feishu.cn/wiki/MsB7wLebki07eUkAZ1ec12W3nsh 一、简介 IIC协议,又称I2C协议,是由PHILP公司在80年代开发的两线式串行总线,用于连接微控制器及其外围设备,IIC属于半双工同步通信方式。 IIC是一种同步…...

Payload的定义及核心概念

在IT领域,Payload(有效载荷) 指数据传输或操作中承载实际功能或信息的主体部分,与协议头、元数据等辅助内容区分。其核心特点是完成特定目标,例如传递关键数据、执行代码逻辑或实现攻击行为。 主要应用场景及技术解析 …...

idea插件使用

文章目录 在哪里安装插件常用插件 在哪里安装插件 离线下载插件请参考我的文章–>可复用性代码 在线下载 插件 :File(文件) —> Settings(设置) —> Plugins(插件) 常用插件 汉化插件:Chinese (Simplified) Language Pack /中文语言包。作者:…...

计算机网络笔记(二十四)——4.6互联网的路由选择协议

4.6.1有关路由选择协议的几个基本概念 路由选择协议是计算机网络中维护和生成路由表的核心机制。 1. 路由选择的核心目标 转发(Forwarding):路由器基于本地转发表,将分组从输入链路转移到输出链路(单台路由器的本地…...

UniApp 微信小程序绑定动态样式 :style 避坑指南

在使用 UniApp 开发跨端应用时,绑定动态样式 :style 是非常常见的操作。然而,很多开发者在编译为 微信小程序 时会遇到一个奇怪的问题: 原本在 H5 中可以正常渲染的样式,在微信小程序中却不生效! 让我们通过一个示例来…...

论文阅读与写作:《从探索到突破:解密科研和论文写作的思维密码》

文章目录 一、如何做科研1.科研的步骤2.课题选择3.快速入门一个新领域:读论文,先读综述(1)自己看论文的时候,每篇论文花3-5分钟记录一下自己的idea和一些瞬间的想法(2)高质量文献:顶会顶刊(3)如何检索 4.注重团队协作与学术交流5.…...

致远OA绩效考核管理应用【附百度网盘下载链接,官方售价8K】

产品概述 绩效考核管理预置三种考核方式:工作事务考核、关键绩效考核、360度考评,满足不同企业考核需求,从考核等级定义、考核方案设置、考核分发到员工考核,再到考核结果汇总并审批,对绩效考核全过程进行闭环管理&…...

Vue百日学习计划Day4-8——Gemini版

番茄时钟: 每个番茄钟为25分钟学习,之后休息5分钟。每完成4个番茄钟,进行一次15-30分钟的长休息。灵活性: 这仍然是一个建议性计划。某些主题(尤其是 Flexbox 和 Grid)可能需要比预期更多的时间来练习和理解…...

HarmonyOS NEXT 适配高德地图FlutterSDK实现地图展示,添加覆盖物和移动Camera

HarmonyOS NEXT 适配高德地图 Flutter SDK 实现地图展示,添加覆盖物和移动 Camera 在现代移动应用开发中,地图功能是许多应用的核心组成部分之一。HarmonyOS NEXT 提供了强大的跨平台开发能力,而高德地图 Flutter SDK 则为开发者提供了丰富的…...

DeepSeek执行流程加速指南:跨框架转换与编译优化的核心策略全解析

💝💝💝欢迎来到我的博客,很高兴能够在这里和您见面!希望您在这里可以感受到一份轻松愉快的氛围,不仅可以获得有趣的内容和知识,也可以畅所欲言、分享您的想法和见解。 推荐:kwan 的首页,持续学…...

docker-compose——安装redis

文章目录 一、编写docker-compose.yaml文件二、编写redis.conf文件三、启动docker-compose 一、编写docker-compose.yaml文件 version: 3.3 services:redis:image: redis:latestcontainer_name: redisrestart: alwaysports:- 6379:6379volumes:- ./redis/data:/data- ./redis/…...

manuskript开源程序是面向作家的开源工具

一、软件介绍 文末提供程序和源码下载 manuskript开源程序是面向作家的开源工具,Manuskript 可在 GNU/Linux、Mac OS X 和 Windows 上运行。 二、Features 特征 Manuskript provides a rich environment to help writers create their first draft and then furt…...

游戏引擎学习第281天:在房间之间为摄像机添加动画效果

回顾并为今天的内容定下基调 这次我们要继续深入处理实体系统。在前一阶段对实体系统做了一些很酷的改动,但现在到了要认真面对和完善它的时候。 今天的主要目标是修复并优化摄像机在房间之间移动时的逻辑。在上一次的实现中,我们重新启用了基于房间的…...

Kaamel隐私合规洞察:Temu在韩被罚事件分析

Kaamel隐私合规与数据安全团队分析报告 韩国个人信息保护委员会(PIPC)对中国电子商务巨头Temu处以巨额罚款,原因是其严重违反了用户数据保护法律 。核心违规行为包括未经适当披露或用户同意非法跨境传输数据、未能指定当地代表、账户注销流程…...

计算机视觉----基于锚点的车道线检测、从Line-CNN到CLRNet到CLRKDNet 本文所提算法Line-CNN 后续会更新以下全部算法

本文所提算法如下: 叙述按时间顺序 你也可以把本文当作快速阅读这几篇文献的一个途径 所有重要的部分我都已经标注并弄懂其原理 方便自己也是方便大家 Line-CNN:基于线提议单元的端到端交通线检测 摘要 交通线检测是一项基础且具有挑战性的任务。以往的…...

25.5.15

没有比水题更令人开心的事情了 典型的并查集题目,并查集分为并和查,并就是把有关系的父亲根结点设为同一个,查就是在成功构造后对其进行查询 查通过递归实现 if (x f[x])return x; return f[x] find(f[x]); 由于并查集的特点&#xff0…...

5.重建大师数据管理模块介绍

摘要:本文主要介绍重建大师数据管理模块,包含:照片、点云数据可视化管理工具。 数据管理界面主要包含工具栏、可视化界面和照片组列表三部分。 图 数据管理界面 1.工具栏 工具栏包含以下功能按钮,包含添加照片、视频、点云、控制…...

MATLAB安装常见问题及解决方案详解(含代码示例)

MATLAB作为科学计算和工程分析的核心工具,其安装过程可能因操作系统版本、硬件配置或网络环境等因素而出现各种问题。本文基于MATLAB官方文档和社区经验,系统总结了安装过程中常见的问题,并提供详细的解决方案和代码示例,帮助用户…...

微信小程序智能商城系统(uniapp+Springboot后端+vue管理端)

一、系统介绍 本智能商城系统是基于当今主流技术栈开发的一款多端商城解决方案,主要包括微信小程序前端、SpringBoot 后端服务以及 Vue 管理后台三大部分。系统融合了线上商城的核心功能,支持商品浏览、下单、支付、订单管理等操作,适用于中小…...

【Spark分析HBase数据】Spark读取并分析HBase数据

Spark读取并分析HBase数据 一、摘要二、实现过程三、小结 一、摘要 Apache Spark 是一个快速、通用的大数据处理引擎,提供了丰富的 API 用于数据处理和分析。HBase 是一个分布式、可扩展的 NoSQL 数据库,适合存储海量结构化和半结构化数据。Spark 与 HB…...

大数据Flink相关面试题(一)

文章目录 一、基础概念‌1. Flink的核心设计目标是什么?与Spark Streaming的架构差异?2. 解释Flink的“有状态流处理”概念。3. Flink的流处理(DataStream API)与批处理(DataSet API)底层执行模型有何不同&…...

填坑记: 古董项目Apache POI 依赖异常排除

当你看到NoSuchMethodError的时候,不要慌,深呼吸,这可能只是JAR包版本的问题… 引子:一个平静的周二下午 那是一个看似平常的周二下午,系统运行良好,开发团队在有条不紊地推进着新功能的开发。突然&#x…...

leetcode2934. 最大化数组末位元素的最少操作次数-medium

1 题目:最大化数组末位元素的最少操作次数 官方标定难度:中 给你两个下标从 0 开始的整数数组 nums1 和 nums2 ,这两个数组的长度都是 n 。 你可以执行一系列 操作(可能不执行)。 在每次操作中,你可以选…...

环境配置与MySQL简介

目录 1 环境配置 2 MySQL简介 1 环境配置 本专栏使用CentOS7进行讲解。首先我们查看系统中是否已经安装了MySQL,可以使用rpm -qa 命令查看系统安装包/压缩包 列表 这只是看我们是否下载过对应安装包,不一定就安装了。如果我们需要重新下载,…...