JVM虚拟机的组成

一、为什么要学习 JVM ?

1. “ ⾯试造⽕箭,⼯作拧螺丝” , JVM 属于⾯试官特别喜欢提问的知识点;

2. 未来在⼯作场景中,也许你会遇到以下场景:

  • 线上系统突然宕机,系统⽆法访问,甚⾄直接OOM
  • 线上系统响应速度太慢,优化系统性能过程中发现 CPU占用过高,原因也许是因为 JVM GC 次数过于频繁;
  • 新项⽬上线,需要设置 JVM 的各种参数;

二、JDK\JRE\JVM

  • JDK:全称“Java Development Kit ” Java 开发⼯具包,提供 javac 编译器、 jconsole,jheap 等监控⼯具;
  • JRE:全称“ Java Runtime Environment ” Java 运⾏环境,提供 Class Library 核心类库+JVM;
  • JVM:全称“ Java Virtual Machine ” Java 虚拟机,⽤于运⾏ Java 应⽤程序; 

三、Java程序执⾏过程

  1. 编译:通过 件。 javac 命令,调⽤ JDK 编译器,将
  2. 执⾏:通过 *.java 源⽂件编译成 *.class 字节码⽂ java 命令,调⽤ JVM 虚拟机,执⾏ *.class 字节码⽂件。

四、Java程序的跨平台 

计算机平台是什么?       

        我们通常把 CPU 处理器与操作系统构成的系统架构,称为计算机平台。

        现代的电⼦计算机是基于⼆进制设计实现,所以 CPU 仅能识别 0 与 1 这样的⼆进制信号。⽽ 计算机程序的本质就是 0 与 1 之间的不同组合产⽣的机器指令,交给 CPU 去执⾏。CPU为了能识 别执⾏机器指令,就需要不同 CPU 架构和指令集来⽀持。不同的⼚商,设计⽣产 的 CPU , CPU 架构和指令集也是不同的。

        常⻅的指令集主要分为:精简指令集( RISC )和复杂指令集( CISC )。

  • 复杂指令集( CISC ): X86 和 X64 两种 CPU 架构基于 CISC 复杂指令集,⽐如: Intel 、 AMD 等⼚商的 CPU 。
  • 精简指令集( RISC ): ARM 架构的CPU基于 RISC 精简指令集,⽐如: Compaq 的 Alpha 、 HP 的 PA-RISC 、 IBM 的 Power PC 、 Apple 的 M1 。 在底层硬件及指令集之上,需要搭载不同的操作系统,⽤于⽀持不同的 CPU 指令集。

        例如:早 期的 MacOS 操作系统只⽀持 Power 系列的 CPU 。最新的 MacOS 系统只能安装在 M1 芯⽚的 CPU 。

        Java 是⼀种跨平台的编程语⾔,主要是为了让相同的 Java 应⽤程序代码,不⽤做任何修 改,可以在不同计算机系统平台上正常运⾏,兼容各种主流操作系统+ CPU 指令集。


跨平台的原因

实现跨平台的原因,主要有两个

  1. 不同的平台,相同的源代码,编译的字节码是相同的,所以字节码⽂件可以在不同平台“ 通⽤” ;
  2. 不同的平台,执⾏字节码时,都需要各⾃的 JVM 虚拟机版本,⽤于将字节码翻译成当前平台可 以执⾏的机器码指令;

五、静态编译器

通过 JDK 提供的静态编译器,将Java 源⽂件编译成字节码⽂件。

编译过程包括:

  1. 词法分析:通过空格分隔出源代码中的单词、操作符、控制符等信息,将其形成 流,传递给语法解析器。
  2. 语法分析:将 token 信息流按照 Java 语法规则,组装成⼀颗语法树
  3. 语义分析:对语法树进⾏关键字使⽤、类型匹配、作⽤域等语法检查。
  4. 字节码:当语义分析完成后,可以⽣成字节码

六、字节码

        当1个 Java 源文件被编译后,就会按照 JVM 规范和字节码定义,产生 1个 class 字节
码文件,文件内容由 10 个固定部分组成。

        前 4 个字节非常特殊,红色框的 cafe babe 是詹姆斯定义的魔法数,它的作用是标志当前文件是一个字节码文件。当 JVM 在进行类加载的 Load 阶段时,如果没有识别到该标志,就说明文件不是字节码文件或已损坏,则无法进行加载。绿色框代表当前版本号,0x34 的十进制为 52 ,是JDK8 的内部版本号,代表这个字节码文件基于 JDK8 编译。
        剩余部分中的每个字节均代表一个字节码指令,由于每个指令的长度按照 1 个字节存储,所以每个指令被称为字节码(Bytecode ),Java 所有的字节码指令有 200 多个。


        由于纯数字的字节码指令阅读⽐较困难,所以 JVM 在字节码指令的基础上设计了⼀套操作码助 记符,使⽤特殊单词来代替对应的数字指令。

例如:

  • 0x15 ILOAD :读取 int 类型的局部变量
  • 0x36 ISTORE :保存int类型的局部变量
  • 0x60 IADD :执⾏两个 int 类型的数值加法运算
  • 0xbb NEW :创建对象
  • 0xbc NEWARRAY :创建数组
  • 0xac IRETURN :返回 int 类型结果

七、JVM执⾏⽅式

JVM执⾏⽅式

JVM 以解释+编译混合模式,执⾏字节码⽂件。

  1. JVM 的执⾏⽅式以解释执⾏为主。执⾏过程中, JVM 将每个字节码⽂件中的每个指令,通过 解释器转换成当前平台可以识别的机器码,然后交给CPU 执⾏。
  2. 为了提⾼执⾏效率, JVM 还会在运⾏期间, JVM 通过热点代码的统计分析,识别⾼频的⽅法 调⽤,循环体、公共模块等,当超过阈值时, JVM 会基于 JIT 即时编译器( me compiler )将热点代码转换成机器码,直接交给 just-in-ti CPU 执⾏,提⾼执⾏效率。

a. Client 模式下默认阈值是 1500 次, Server 模式下是 10000 次。

  • ● JVM 解释器:程序执⾏的时候,解释器⾸先发挥作⽤,省去了编译器编译时间,加快程序的执 ⾏效率。
  • ● JIT 编译器:在程序运⾏过程中,随着时间的推移, JIT 开始慢慢发挥了作⽤,把热点代码 编译成本地代码后,以后执⾏相同的代码,即可直接交给 CPU 执⾏,带来更⾼的执⾏效率。

        机器在热机状态可以承受的负载要⼤于冷机状态(刚启动时 ),如果以热机状态时的流量进⾏切 换,可能使处于冷机状态的服务器因⽆法承载流量⽽假死。所以,在⽣产环境中发布应⽤,应该以分 批的⽅进⾏发布,根据机器数量划分成多个批次,每个批次的机器数⼤概占到整个集群的 15% 。

        故障案例:某应⽤在线发布新版本,采⽤进⾏分批发布,发布总批数误填写成分为两批发布。如 果是热机状态,正常情况下,集群中的⼀半的机器可以勉强⽀撑负载流量,但由于刚启动的 JVM 均 是解释执⾏,还没有进⾏热点代码统计和JIT动态编译,导致机器启动之后,当前 1/2 发布成功的 服务器⻢上全部宕机。

八、类加载器 

        字节码必须通过类加载器,通过加载、验证、解析等校验步骤,将字节码⽂件中的类,加载⾄ JVM 的中运⾏时数据区,才可以执⾏字节码。

类加载器的加载过程

当一个类被使用的时候,才会加载到内存中,类加载的过程有:加载、验证、准备、解析、初始化。其中中间三步合并称为链接。

  • - 加载:通过包名+类名,获取这个类,准备用流进行传输,将这个类加载到内存中,并创建一个class对象
  • - 验证:验证class文件中的信息是否符合虚拟机规范,有没有安全隐患
  • - 准备:负责为类的类变量(static修饰的变量)分配内存,并设置默认的初始化值
  • - 解析:将类的二级制数据流中的符号引用替换为直接引用。(本类中如果用到了其他类,此时就需要找到对应的类)
  • - 初始化:静态变量赋值以及初始化其他资源

九、垃圾收集器

        JVM 在运⾏期间,通过 Garbage Collctor 垃圾收集器,定期对运⾏时数据区进⾏垃圾对 象的回收,从⽽实现了内存⾃动管理。

十、JVM组成结构

        JVM 由类加载器、运⾏时数据区、 JVM 解释器、 JIT 即时编译器、垃圾回收器、本地⽅法库等部分组成。

        由类加载器完成字节码⽂件的加载验证和解析,存储⾄运⾏时数据区,并由执⾏引擎中的解释 器,完成字节码到机器码的解释执⾏。同时进⾏热点代码的统计分析,调⽤ JIT 即时编译器将字节 码直接编译成机器码,提⾼执⾏效率。JVM运⾏期间的⽅法调、数据对象统⼀存放⾄运⾏时数据区。

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

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

相关文章

2024年虚拟现实、图像和信号处理国际学术会议(ICVISP 2024,8月2日-4)

2024年虚拟现实、图像和信号处理国际学术会议(ICVISP 2024)将于2024年8月2-4日在中国厦门召开。ICVISP 2024将围绕“虚拟现实、图像和信号处理”的最新研究领域, 为来自国内外高等院校、科学研究所、企事业单位的专家、教授、学者、工程师等提…

迁移方案详解|使用YMP从异构数据库迁移到YashanDB

数据迁移简介 01典型场景与需求 在国产化浪潮下,数据库系统的国产化替代成为了一个日益重要的议题,有助于企业降低对外依赖,提升信息安全和自主性。 以Oracle、MySQL为代表的传统关系型数据库管理系统,在企业应用中占据了重要的…

图书馆借阅表

DDL 用户表 (Users) 图书表 (Books) 图书类别表 (BookCategories) 图书与类别关联表 (BookCategoryRelations) 借阅记录表 (BorrowRecords) 供应商表 (Suppliers) 采购记录表 (PurchaseRecords) CREATE TABLE Users (user_id INT PRIMARY KEY AUTO_INCREMENT,username …

pytorch神经网络训练(VGG-19)

VGG-19 导包 import torchimport torch.nn as nnimport torch.optim as optimimport torchvisionfrom torchvision import datasets, transformsfrom torch.utils.data import DataLoaderimport matplotlib.pyplot as plt 数据预处理和增强 transform transforms.Compose(…

在 Go 中如何让结构体不可比较?

最近我在使用 Go 官方出品的结构化日志包 slog 时,看到 slog.Value 源码中有一个比较好玩的小 Tips,可以限制两个结构体之间的相等性比较,本文就来跟大家分享下。 在 Go 中结构体可以比较吗? 在 Go 中结构体可以比较吗&#xff…

鸿蒙开发HarmonyOS NEXT(一)

最近总听见大家讨论鸿蒙,前端转型的好方向?先入门学习下 目前官方版本和文档持续更新中 一、开发环境 提示:要占用的空间比较多,建议安装在剩余空间多的盘 1、下载:官网最新工具 - 下载中心 - 华为开发者联盟 (huaw…

RTL8305NB从电口模式切换为光口模式

对于RTL8305NB,要从电口模式切换为光口模式,主要操作涉及到PHY page的切换和特定寄存器的配置。以下是详细的操作步骤: PHY Page切换: 首先,需要访问PHY地址8的寄存器31。这个寄存器用于Page的切换。向PHY地址8的寄存…

从删库到还原

欢迎来到我的博客,代码的世界里,每一行都是一个故事 🎏:你只管努力,剩下的交给时间 🏠 :小破站 从删库到还原 魔法一魔法二魔法三魔法四查看是否开启binlog,且format为row执行以下命…

WAV怎么转mp3?将wav转成MP3的几种方法介绍

WAV怎么转mp3?很多情况下,我们可能需要将高质量的 WAV 文件转换为更小、更兼容的 MP3 文件。例如,你可能想要为你的音乐收藏腾出更多存储空间,或者需要将音频文件上传到联网平台,而这些平台通常对文件大小有严格限制。…

会声会影2024免费版下载无需激活码序列号

亲爱的影像爱好者们,今天我要和大家分享的是一款让我彻底着迷的软件——会声会影2024!自从用了它,我的视频编辑技能简直突飞猛进,每次上传作品到小红书都能收获满满的赞👍。接下来,就让我带你一起探索这个神…

window系统忘记密码解决方案

原理 通过命令修改粘滞键的作用打开cmd命令,通过cmd命令修改用户密码。 1.进入系统自动恢复页面 各品牌进入恢复页面各不一样,一般按住shift重启电脑即可,笔者的惠普电脑是开机按住F11键。页面如下: 之后选择 - > 疑难解答…

阿里云nginx更新证书后依旧显示旧证书

尝试的解决办法 重启nginx服务删除服务器上的旧证书清除浏览器缓存检查是否使用CDN服务 最后的解决办法 云服务器开启了WAF服务,在WAF服务中配置证书

ssm 宠物领养系统-计算机毕业设计源码08465

目 录 摘要 1 绪论 1.1课题背景及意义 1.2研究现状 1.3ssm框架介绍 1.3论文结构与章节安排 2 宠物领养系统系统分析 2.1 可行性分析 2.2 系统流程分析 2.2.1 数据流程 3.3.2 业务流程 2.3 系统功能分析 2.3.1 功能性分析 2.3.2 非功能性分析 2.4 系统用例分析 …

web开发学习(web简单入门)

前言: 从我刚接触博客没多久我就萌发了搭建一个个人博客网站的想法(用来装逼),但碍于学校屁事太多迟迟没有开始,最近学校课已经都差不多结课了,距离期末还有一段时间,我也得以抽出时间来学习我一…

js实现blockly后台解释器,可以单步执行,可以调用c/c++函数

实现原理 解析blockly语法树,使用js管理状态,实际使用lua执行,c/c++函数调用使用lua调用c/c++函数的能力 可以单行执行 已实现if功能 TODO for循环功能 函数功能 单步执行效果图 直接执行效果图 源代码 //0 暂停 1 单步执行 2 断点 //创建枚举 var AstStatus = {PAUS…

35、正则表达式

一、正则表达式命令 正则表达式:匹配的是文本内容,linux的文本三剑客都是针对文本内容。 ​ grep 过滤文本内容 ​ sed 针对文本内容进行增删改查 ​ awk 按行取列 文本三剑客----都是按照行进行匹配。 1.1、grep筛选: grep的作用就是…

私有化地图离线部署方案之查询定位服务

私有化地图离线部署整体解决方案,除硬件之外,一般主要由基础地图服务、查询定位服务、路径规划服务和高程检索服务构成。 其中,查询定位服务是指地理编码与逆地理编码服务。 在《私有化地图离线部署方案之基础地图服务》一文中,…

不懂电路搭建可以学嵌入式编程开发吗?

当然可以学嵌入式编程开发!虽然电路搭建是嵌入式开发中的一部分,但即使你对电路搭建不太了解,也可以从嵌入式编程开发入手。刚好我有一些资料,是我根据网友给的问题精心整理了一份「嵌入式的资料从专业入门到高级教程」&#xff0…

宝塔面板部署前端项目

部署前端项目 1 打包自己的项目2 登录宝塔面板3 添加站点4 设置域名5 进入当前站点对应的文件目录中6 上传打包后的文件7 访问网站 1 打包自己的项目 2 登录宝塔面板 点击左侧“网站”菜单进入对应页面 点击“添加站点” 3 添加站点 填写域名,如果没有域名的&am…

集成openfeign

集成feign有两种方式. 1.集成到所需项目中(只有该项目可以用)直接引用所需调用的项目 2.集成到公共项目(通用) 1.集成到所需项目中(只有该项目可以用) 再需要消费的服务 进行依赖的引用 1.引入依赖,openfeign,和所需调用的服务 <!--feign--><dependency><gro…