【从零开始学习JVM | 第二篇】字节码文件的组成

前言:

        字节码作为JAVA跨平台的主要原因,熟练的掌握JAVA字节码文件的组成可以帮助我们解决项目的各种问题,并且在面试中,关于字节码部分的内容却是一大考点和难点,因此我们在这里穿插讲解一下字节码文件的组成。

目录

前言:

1.如何打开字节码文件? 

2.字节码文件的组成:

3.简单阅读字节码

总结:


 

1.如何打开字节码文件? 

字节码文件中保存了原代码编译后的内容,以二进制的形式进行存储。因此我们是无法用记事本这种普通的阅读软件打开的。

因此我们要使用专业的字节码阅读工具进行。好在IDEA中就有对应的插件,我们只需要点击下载就可以。

jclasslib Bytecode Viewer

 然后通过这个插件,我们就可以便捷的查看一个字节码文件:

我们来逐一解释一下这些部分:

1.基础信息:魔数,字节码文件对应的JAVA版本号,访问标识符 ,父类和接口。

2.常量信息:保存了字符串常量,类或接口名,字段名。主要在字节码指令中使用。

3.字段:当前类或接口声明的字段信息。

4.方法:当前类或接口声明的方法信息字节码指令。

5.属性 :类的属性,比如源码的文件名,内部类的列表等。 

通过这款插件,我们就可以看到上图的字节码:

2.字节码文件的组成:

ava字节码文件(通常以.class扩展名结尾)是Java源代码编译后的产物。它包含了Java程序在JVM上执行所需的指令和信息。字节码文件的组成结构非常标准化,按照特定的格式排列,主要由以下部分组成:

  1. 魔数(Magic Number)

    • 每个字节码文件的前4个字节都是固定的魔数0xCAFEBABE,用于标识这是一个Java字节码文件。
  2. 版本信息

    • 紧接着魔数的是版本信息,包括次版本号(Minor Version)和主版本号(Major Version),用于表示编译该字节码的Java编译器版本。
  3. 常量池(Constant Pool)

    • 常量池紧随版本信息之后,是.class文件中资源的集合,包括各种文字字符串、类或接口名、字段名和其他常量。
    • 它是一个表结构,存储了类中所有的符号引用,这些引用将在类加载阶段被解析。
  4. 访问标志(Access Flags)

    • 用于描述类或接口的访问权限(如public、private、protected)和属性(如abstract、final、interface等)。
  5. 此类索引、父类索引和接口索引集合

    • 这些索引值指向常量池中的项,分别代表当前类、超类和实现的接口的符号引用。
  6. 字段表(Fields Table)

    • 包含类或接口中声明的所有字段,每个字段都有其对应的属性集合,如名称、类型、访问标志等。
  7. 方法表(Methods Table

    • 列出了类中的所有方法,包括方法的名称、返回类型、参数类型、访问标志以及方法的字节码指令。
  8. 属性表(Attributes Table)

    • 属性表为字段表、方法表提供附加信息,如异常表、行号表、局部变量表等。每个属性都有自己的结构定义,例如,Code属性就包含了Java方法的JVM指令、操作数栈、局部变量表等信息。
  9. 接口表(Interfaces Table)

    • 列出该类实现的所有接口。

字节码文件的这些组成部分共同定义了Java类的结构和行为。当JVM加载一个类时,它会解析字节码文件并根据其中的信息创建出相应的Class对象,然后再在JVM上执行该类的代码。字节码文件的结构设计使得Java程序具有很强的跨平台性,可以在任何安装了兼容JVM的设备上运行。

很多同学都不理解这个魔数,因此我们换种说法解释一下:由于我们可以随意的更改一个文件的后缀,例如把jpg更改为png。因此电脑是无法通过文件扩展名来确定文件类型的。因此如电脑需要通过文件的头几个字节(文件头)去校验文件的类型。而在JAVA字节码文件中,将文件头叫做魔数 

3.简单阅读字节码

 了解了字节码文件的组成部分之后,我们来看看一段字节码以及解释:

 0 iconst_0          // 将int类型常量0压入操作数栈顶
 1 istore_1          // 将操作数栈顶的int类型数值(0)存入第二个局部变量槽中(局部变量索引1)
 2 iload_1           // 从局部变量表中加载索引为1的int类型值到操作数栈顶
 3 iinc 1 by 1       // 将局部变量表中索引为1的int类型变量增加1
 6 istore_1          // 将操作数栈顶的int类型数值(经过iinc后的值)存入第二个局部变量槽中(局部变量索引1)
 7 getstatic #2 <java/lang/System.out : Ljava/io/PrintStream;> // 获取类java.lang.System的静态字段out的值,即PrintStream对象,并压入操作数栈顶
10 iload_1           // 从局部变量表中加载索引为1的int类型值到操作数栈顶
11 invokevirtual #3 <java/io/PrintStream.println : (I)V> // 调用PrintStream对象的println方法打印int值
14 return            // 从当前方法返回

 这段字节码对应的代码就是:

看不懂的同学们对着我的字节码注释自己尝试一步一步走一下就理解了。其实我们通过字节码就可以清晰的知道为什么代码运行结果等于0。

我们再来看一眼这个代码的:

他的字节码指令为:

 0 iconst_0
 1 istore_1
 2 iinc 1 by 1
 5 iload_1
 6 istore_1
 7 getstatic #2 <java/lang/System.out : Ljava/io/PrintStream;>
10 iload_1
11 invokevirtual #3 <java/io/PrintStream.println : (I)V>
14 return

由此我们可以看出:之所以i++和++i不一样,从字节码角度上来看,是因为iinc和iload的执行顺序不一样,导致一个在打印的时候加载的是旧值,一个是新值。

总结来讲:int i=0; i = i ++之所以结果等于0,是因为在字节码中,i ++ 先把 0 取出来放入到临时的操作数栈中,接下来对 i 进行加 1 操作,i 变为了1,最后再把操作数栈中之前保存的i值拿出来放入i中,使得i变为了0。

而int i=0; i = ++ i 之所以结果等于 1 ,是因为在字节码中,++i 先进行自增,使得i变为 1 ,然后再将 1 取出来放入到临时的操作数栈中,再把操作数栈中的1拿出来赋值为i,使得i变为了 1 .

总结:

当我们编写和阅读Java代码时,我们通常关注的是高级语言层面的代码。然而,在Java编译器将高级代码转换为可执行的机器码之前,它会将我们的代码转换为一种称为字节码的中间表示形式。

字节码是一种与特定平台无关的指令集,它使用单字节的操作码和操作数来描述操作。它是Java虚拟机(JVM)的基本指令集,JVM可以解释执行字节码或将其编译为机器码。

通过深入理解字节码的组成,我们可以更好地理解Java代码的底层运行原理,进一步优化性能和调试问题。

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

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

[跑代码-遇到问题-报错3]BK-SDM. KeyError: ‘up_blocks.0‘

File "src/kd_train_text_to_image.py", line 790, in mainKeyError: up_blocks.0 出问题的原因 dict acts_tea读到dict acts_stu没有读到dict 原因是 unet_teacher的结构后面直接接down_blocks&#xff08;正常&#xff09;unet_teacher.down_blocks 但是unet的结构…

java:springboot3集成swagger(springdoc-openapi-starter-webmvc-ui)

背景 网上集成 swagger 很多都是 Springfox 那个版本的&#xff0c;但是那个版本已经不更新了&#xff0c;springboot3 集成会报错 Typejavax.servlet.http.HttpServletRequest not present&#xff0c;我尝试了很多才知道现在用 Springdoc 了&#xff0c;今天我们来入门一下 …

对标Gen-2!Meta发布新模型进军文生视频赛道

随着扩散模型的飞速发展&#xff0c;诞生了Midjourney、DALLE 3、Stable Difusion等一大批出色的文生图模型。但在文生视频领域却进步缓慢&#xff0c;因为文生视频多数采用逐帧生成的方式,这类自回归方法运算效率低下、成本高。 即便使用先生成关键帧,再生成中间帧新方法。如…

aikit 2023 3D与机械臂结合!

引言 今天我们主要了解3D摄像头是如何跟机械臂应用相结合的。我们最近准备推出一款新的机械臂套装AI Kit 2023 3D&#xff0c;熟悉我们的老用户应该知道&#xff0c;我们之前的AI Kit 2023套装使用的是2D摄像头。 随着技术进步&#xff0c;市场需求和领域的扩大&#xff0c;2D的…

第一百九十回 自定义一个可选择的星期组件

文章目录 1. 概念介绍2. 实现方法2.1 实现思路2.2 实现方法3. 示例代码4. 内容总结我们在上一章回中介绍了"如何让Text组件中的文字自动换行"相关的内容,本章回中将介绍 如何自定义一个可选择的星期组件.闲话休提,让我们一起Talk Flutter吧。 1. 概念介绍 我们在…

在Word中移动页面主要靠导航窗格,有了它,移动页面就事半功倍

本文包括有关在Microsoft Word 2019、2016和Office 365中使用导航窗格移动页面以及复制和粘贴页面的说明。 如何设置导航窗格以重新排列页面 Microsoft Word并不将文档视为单独页面的集合,而是将其视为一个长页面。正因为如此,重新排列Word文档可能会很复杂。在Word中移动页…

C++ 操作MinIO做文件数据的上传和下载(踩坑与经验)包含编译包

前言 最近在做项目流程优化&#xff0c;准备将之前的java对文件的操作转换到c端&#xff0c;因此做了基于c的minio操作的测试demo。期间的各种踩坑与问题&#xff0c;花了一天时间总算是成功了&#xff0c;当然还有一些小问题&#xff0c;等待后续其他大拿解决。 项目环境 v…

linux 中crontab 定时任务计划创建时间文件夹示例

1.创建一个sh脚本 /usr/bin/mkdir 是mkdir命令的路径 /usr/bin/chmod 是chmod命令的路径 2.编辑定时任务 crontab -e

Hadoop学习笔记(HDP)-Part.11 安装Kerberos

目录 Part.01 关于HDP Part.02 核心组件原理 Part.03 资源规划 Part.04 基础环境配置 Part.05 Yum源配置 Part.06 安装OracleJDK Part.07 安装MySQL Part.08 部署Ambari集群 Part.09 安装OpenLDAP Part.10 创建集群 Part.11 安装Kerberos Part.12 安装HDFS Part.13 安装Ranger …

最新测试开发招聘信息汇总,内含社招和实习生岗位~

1 科大讯飞 [武汉/合肥/西安] 科大讯飞&#xff08;教育事业部&#xff09; - 移动、服务端高级测试开发工程师 一、移动方向高级测试开发工程师 岗位职责&#xff1a; 1.负责教育 BG 中 APP/SDK/软硬一体等产品类型的专项测试工作&#xff0c;包括专项测试方案设计、自动化测…

csp 现值计算 C语言

号&#xff1a; 202212-1 试题名称&#xff1a; 现值计算 时间限制&#xff1a; 1.0s 内存限制&#xff1a; 512.0MB 问题描述&#xff1a; 问题描述 评估一个长期项目的投资收益&#xff0c;资金的时间价值是一个必须要考虑到的因素。简单来说&#xff0c;假设…

背包问题学习

背包问题是常见的动态规划dp的问题 下面用到的符号: 常用n表示物品数, m表示背包容积f[i][j]表示i件物品, j的背包容量的最大价值w[i]表示第i件物品的价值, v[i] 表示第i件物品的容量f[0][0~m] 0, 所以n可以从1开始遍历一般是有两层嵌套循环 第一层遍历物品, 第二层遍历背包…

计网Lesson6 - IP 地址分类管理

文章目录 1. I P IP IP 地址定义2. I P v 4 IPv4 IPv4 的表示方法2.1 I P v 4 IPv4 IPv4 的分类编址法2.2 I P v 4 IPv4 IPv4 的划分子网法2.2.1 如何划分子网2.2.2 如何确定子网的借位数2.2.3 总结2.2.4 题目练习 2.3 I P v 4 IPv4 IPv4 的无分类编址法 1. I P IP IP 地…

zabbix的自动发现机制、代理功能、SNMP监控

一、自动发现&#xff08;不安全&#xff0c;有时会失效&#xff0c;建议手动添加主机&#xff09; 1、定义 zabbix主动与服务端联系&#xff0c;将自己的地址和端口发送给服务端&#xff0c;实现自动添加监控主机 客户端是主动的一方 2、缺点 若自定义网段中主机数量太多…

电商API接口开发和接入说明{包含淘宝/京东/拼多多/抖音}

“为什么改了这个没告诉我” “实际功能和文档上说的不一样啊”。 这些话大家在进行电商API接口开发时&#xff0c;想必耳朵都听出老茧了。 真不是故意的&#xff0c;有时候任务比较急&#xff0c;就先改了代码&#xff0c;想着以后再同步文档&#xff0c;然后就给忘了。 项…

HarmonyOS带大家创建自己的第一个Page页面并实现路由跳转

我们 在开发过程中 经常会看到 被 艾特修饰的代码 有限像 java中的注解 在 harmonyOS 中 这叫 装饰器 被关键字装饰取来的代码 会具备某某功能 我们这里先来创建一个新的界面 在pages 目录下 右键 如下图 选择page创建 这里 我们取名叫 AppView 然后点击右下角 Finish 这样…

线程池原理初探

1.引言 合理利用线程池能够带来三个好处。第一&#xff1a;降低资源消耗。通过重复利用已创建的线程降低线程创建和销毁造成的消耗。第二&#xff1a;提高响应速度。当任务到达时&#xff0c;任务可以不需要的等到线程创建就能立即执行。第三&#xff1a;提高线程的可管理性。…

CoreDNS实战(七)-日志处理

本文主要用于介绍CoreDNS用来记录日志的几种方式以及在生产环境中遇到的一些问题和解决方案。 1 log插件 coredns的日志输出并不如nginx那么完善&#xff08;并不能在配置文件中指定输出的文件目录&#xff0c;但是可以指定日志的格式&#xff09;&#xff0c;默认情况下不论…

手写分析文件大小工具

背景&#xff1a; window 用久了磁盘变红了&#xff0c;又不想安装大文件分析的软件&#xff0c;突发奇想能否自己写一个代码&#xff0c;分析有哪些大文件 文件的单位&#xff0c;最高记作G // 文件大小单位static String[] fileSizeUnits {"B", "KB", …

SpringBoot + Spring Cloud Alibaba + Nacos实现服务管理

1、参考文档 Spring Cloud Alibaba参考文档 https://spring-cloud-alibaba-group.github.io/github-pages/hoxton/zh-cn/index.html Spring Cloud Alibaba官方文档 https://github.com/alibaba/spring-cloud-alibaba/wiki/ 2、引入 Alibaba 依赖 每个 SpringBoot 都有对应的…