JVM 字节码是如何存储信息的?

JVM 字节码是 Java 虚拟机 (JVM) 执行的指令集,它是一种与平台无关的二进制格式,在任何支持 JVM 的平台上都可运行的Java 程序。 字节码存储信息的方式,主要通过以下几个关键组成部分和机制来实现:

1. 指令 (Opcodes) 和 操作数 (Operands):

  • 指令 (Opcodes): 字节码的核心是指令集,每条指令都是一个单字节 (byte) 的数字编码,称为 操作码 (opcode)。 操作码定义了 JVM 需要执行的具体操作,例如:
    • 算术运算: iadd (整数加法), isub (整数减法), fmul (浮点数乘法) 等。
    • 数据加载和存储: iload (加载整数到操作数栈), istore (将整数从操作数栈存储到局部变量表), getfield (获取对象字段值) 等。
    • 类型转换: i2f (整数转浮点数), l2i (长整型转整型) 等。
    • 方法调用: invokevirtual (调用虚方法), invokestatic (调用静态方法), invokeinterface (调用接口方法) 等。
    • 控制流: goto (无条件跳转), ifeq (如果等于 0 则跳转), return (方法返回) 等。
  • 操作数 (Operands): 有些指令需要额外的操作数 (operands) 来指定指令执行所需的参数或数据。 操作数紧跟在操作码之后,可以是:
    • 字节 (byte): 例如,用于表示局部变量表的索引、常量池的索引等。
    • 短整型 (short): 例如,用于表示分支指令的偏移量。
    • 整型 (int): 例如,用于表示常量池的索引 (在某些指令中)。
    • 常量池索引 (constant pool index): 指向常量池中特定项的索引,用于引用类名、方法名、字段名、字符串字面量、数值常量等。

案例:

// Java 源代码
public class Example {public static int add(int a, int b) {return a + b;}
}// 对应的 JVM 字节码 (简化表示)
// 方法 add 的字节码
0: iload_0        // 将局部变量 0 (a) 推入操作数栈
1: iload_1        // 将局部变量 1 (b) 推入操作数栈
2: iadd           // 执行整数加法,栈顶两个值相加,结果推回栈顶
3: ireturn        // 从方法返回,返回栈顶的整数值

在这个例子中:

  • iload_0, iload_1, iadd, ireturn 都是操作码 (指令)。
  • iload_0iload_1 的操作数是隐式的 (隐含了局部变量表的索引 0 和 1)。

2. 常量池 (Constant Pool):

  • 关键的数据结构: 常量池是 .class 文件中的一个表结构,也是字节码存储信息的核心组成部分。 它存储了类、方法、字段、字符串字面量、数值常量等各种符号引用和字面量常量
  • 动态链接的基础: 常量池为 JVM 的动态链接机制提供了基础。 字节码中的指令通常通过常量池索引来引用程序中的各种符号和常量,而不是直接使用内存地址。 这使得字节码具有平台无关性,因为具体的内存地址在运行时才由 JVM 决定。
  • 存储类型: 常量池中的每一项 (constant pool entry) 都有一个 tag 标识其类型,常见的类型包括:
    • CONSTANT_Class_info: 类或接口的符号引用 (类名、接口名)。
    • CONSTANT_Fieldref_info: 字段的符号引用 (类名、字段名、字段描述符)。
    • CONSTANT_Methodref_info: 方法的符号引用 (类名、方法名、方法描述符)。
    • CONSTANT_InterfaceMethodref_info: 接口方法的符号引用。
    • CONSTANT_String_info: 字符串字面量。
    • CONSTANT_Integer_info, CONSTANT_Float_info, CONSTANT_Long_info, CONSTANT_Double_info: 数值常量 (整数、浮点数、长整型、双精度浮点数)。
    • CONSTANT_NameAndType_info: 字段或方法名称和描述符。
    • CONSTANT_Utf8_info: UTF-8 编码的字符串 (用于存储类名、方法名、字段名等字符串)。
    • … (还有其他类型)

例子 (常量池引用):

// Java 源代码
public class Example {private String message = "Hello"; // 字符串字面量 "Hello"public void printMessage() {System.out.println(message); // 引用字段 message}
}// 对应的 JVM 字节码 (简化表示)
// ... (省略其他字节码)
// getfield 指令,操作数是常量池索引 #2
4: getfield      #2  // Field Example.message:Ljava/lang/String;// 常量池 #2 项 (简化表示)
#2 = Fieldref      #4.#5        // 字段引用#4 = Class         #6        // 类名引用#5 = NameAndType   #7:#8      // 名称和类型引用#6 = Utf8          Example     // 类名字符串 "Example"#7 = Utf8          message     // 字段名字符串 "message"#8 = Utf8          Ljava/lang/String; // 字段类型描述符 "Ljava/lang/String;"

在这个例子中:

  • getfield #2 指令使用常量池索引 #2 来引用要访问的字段 message
  • 常量池 #2 项是一个 Fieldref 结构,它又通过其他常量池索引引用了类名、字段名和字段描述符等信息。
  • 字符串字面量 "Hello" 也存储在常量池中 (例如,通过 CONSTANT_String_infoCONSTANT_Utf8_info),并在需要时被引用。

3. 局部变量表 (Local Variable Table):

  • 存储方法内的局部变量: 局部变量表是每个方法在运行时创建的栈帧 (stack frame) 的一部分。 它用于存储方法内部的局部变量,包括:
    • 方法的参数 (arguments)。
    • 方法体内部定义的局部变量。
  • 数组结构: 局部变量表本质上是一个数组,每个数组元素可以存储一个 Java 的基本数据类型值 (int, float, long, double, byte, short, char, boolean) 或对象引用 (reference)。
  • 索引访问: 字节码指令使用索引来访问局部变量表中的变量,例如 iload_0 加载索引为 0 的局部变量,istore_1 将值存储到索引为 1 的局部变量。

4. 操作数栈 (Operand Stack):

  • 运算和操作的工作区: 操作数栈是每个方法栈帧的另一个重要组成部分。 它是一个后进先出 (LIFO) 的栈,用于:
    • 存储指令的操作数: 指令执行时,会从操作数栈中弹出操作数。
    • 存储指令的运算结果: 指令执行完毕后,会将结果压入操作数栈顶。
  • 指令的执行流程: JVM 的字节码指令大多是基于栈的指令集。 指令通常会:
    1. 从局部变量表或常量池加载数据到操作数栈。
    2. 从操作数栈中弹出操作数进行运算。
    3. 将运算结果压入操作数栈。
    4. 将操作数栈顶的值存储到局部变量表或字段中,或者作为方法返回值返回。

5. 方法区 (Method Area) (元空间/永久代):

  • 存储类元数据: 方法区 (在 JDK 8 及之后被元空间 Metaspace 取代,JDK 7 及之前为永久代 PermGen) 用于存储类的信息,包括:
    • 类的结构信息: 例如,类的名称、父类、实现的接口、字段信息、方法信息、访问修饰符等。
    • 运行时常量池 (Runtime Constant Pool): 每个类都有一个运行时常量池,它是 .class 文件常量池在运行时的表示形式,用于支持动态链接。
    • 静态变量: 类的静态变量也存储在方法区中。
    • JIT 编译后的代码: 即时编译器 (JIT Compiler) 编译后的本地机器码通常也存储在方法区中。

总结:

JVM 字节码通过以下方式存储信息:

  • 指令集 (Opcodes): 定义了 JVM 要执行的操作,每条指令都是一个单字节编码。
  • 操作数 (Operands): 为指令提供参数或数据,可以是字节、短整型、整型或常量池索引。
  • 常量池 (Constant Pool): 存储了类、方法、字段、字符串字面量、数值常量等符号引用和字面量常量,是动态链接的基础。
  • 局部变量表 (Local Variable Table): 存储方法内部的局部变量,包括参数和方法体内部定义的变量。
  • 操作数栈 (Operand Stack): 作为指令运算和操作的工作区,存储指令的操作数和运算结果。
  • 方法区 (Method Area) / 元空间 (Metaspace): 存储类的元数据信息,包括类结构、运行时常量池、静态变量等。

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

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

相关文章

基于51单片机语音实时采集系统

基于51单片机语音实时采集 (程序+原理图+PCB+设计报告) 功能介绍 具体功能: 系统由STC89C52单片机ISD4004录音芯片LM386功放模块小喇叭LCD1602按键指示灯电源构成 1.可通过按键随时选择相应的录音进行播…

关于 Java 预先编译(AOT)技术的详细说明,涵盖 GraalVM 的配置、Spring Boot 3.x 的集成、使用示例及优缺点对比

以下是关于 Java 预先编译(AOT)技术的详细说明,涵盖 GraalVM 的配置、Spring Boot 3.x 的集成、使用示例及优缺点对比: 1. 预先编译(AOT)技术详解 1.1 核心概念 AOT(Ahead-of-Time&#xff09…

【ROS2】行为树:BehaviorTree

1、简介 与状态机不同,行为树强调执行动作,而不是状态之间的转换。 行为树是可组合的。可以重复使用简单的行为来构建复杂的行为。 在游戏领域,行为树已经比较流行了。主要用于维护游戏角色的各种动作和状态。 ROS2的导航框架Navigation2中引入了行为树来组织机器人的工作流…

Centos7.9 升级内核,安装RTX5880驱动

系统镜像下载 https://vault.centos.org/7.9.2009/isos/x86_64/CentOS-7-x86_64-DVD-2009.iso 系统安装步骤省略 开始安装显卡驱动 远程登录查看内核 [root192 ~]# uname -a Linux 192.168.119.166 3.10.0-1160.el7.x86_64 #1 SMP Mon Oct 19 16:18:59 UTC 2020 x86_64 x8…

多层感知机与全连接神经网络关系解析

感知机(Perceptron)、多层感知机(MLP,Multilayer Perceptron)和全连接神经网络(FCNN,Fully Connected Neural Network)是神经网络发展过程中密切相关的概念,但它们有明确…

解析医疗器械三大文档:DHF、DMR与DHR

医疗器械的 DHF、DMR 和 DHR 是质量管理体系(QMS)中的核心文件,贯穿产品全生命周期, 确保医疗器械的安全性、有效性和合规性。 一、三大文件的定义与法规依据 缩写全称法规依据(以 FDA 为例)核心目的DHF…

netty启用websocket的压缩机制

netty启用websocket的压缩机制 package com.aerotop.connector.websocket.base;import io.netty.channel.ChannelInitializer; import io.netty.channel.ChannelPipeline; import io.netty.channel.socket.SocketChannel; import io.netty.handler.codec.compression.JZlibDec…

可能存在特殊情况,比如控制台显示有延迟、缓冲问题等影响了显示顺序。

从控制台输出看,正常逻辑应是先执行 System.out.println(" 未处理异常演示 "); 输出对应文本,再因 arr 为 null 访问 length 触发 NullPointerException 输出异常信息。可能存在特殊情况,比如控制台显示有延迟、缓冲问题等影响…

第一节:React 基础篇-React虚拟DOM原理及Diff算法优化策略

必考点:虚拟DOM树对比(同级比较、Key的作用、组件类型判断) 延伸:React 18中并发更新对Diff算法的影响 React虚拟DOM原理及Diff算法优化策略 虚拟DOM核心原理 概念: • 虚拟DOM(Virtual DOM)…

Java spring mybatis面试题(200道),八股文

Java面试题 通过网盘分享的文件:面试题等2个文件 链接: https://pan.baidu.com/s/1Xw0PzkfAmL8uesYBvrW2-A?pwdpebt 提取码: pebt mybatis相关 1、 什么是Mybatis? … 2 2、 Mybaits 的优点: … 2 3、 MyBatis 框架的缺点: ……

windows使用Python调用7-Zip【按大小分组】压缩文件夹中所有文件

使用Python调用7-Zip【按大小分组】压缩文件夹中所有文件 问题描述:方法前提条件任务完整代码示例代码如何工作? 问题描述: 我现在想把文件夹下的所有内容上传到网盘,但是这个文件夹下的素材内容很多,使用分卷压缩的话…

《Python星球日记》第26天:Matplotlib 可视化

名人说:路漫漫其修远兮,吾将上下而求索。—— 屈原《离骚》 创作者:Code_流苏(CSDN)(一个喜欢古诗词和编程的Coder😊) 订阅专栏:《Python星球日记》 目录 一、Matplotlib 简介1. 什么是 Matplo…

第1章 对大型语言模型的介绍

人类正处在一个关键转折点。自2012年起,基于深度神经网络的人工智能系统研发进入快速通道,将这一技术推向了新高度:至2019年底,首个能够撰写与人类文章真假难辨的软件系统问世,这个名为GPT-2(生成型预训练变…

dcsdsds

我将为您在页面顶部添加欢迎内容&#xff0c;同时保持整体风格的一致性。以下是修改后的代码&#xff0c;主要修改了模板部分和对应的样式&#xff1a; vue 复制 <template><div class"main-wrapper"><!-- 新增欢迎部分 --><div class"…

学习MySQL的第八天

海到无边天作岸 山登绝顶我为峰 一、数据库的创建、修改与删除 1.1 引言 在经过前面七天对于MySQL基本知识的学习之后&#xff0c;现在我们从基本的语句命令开始进入综合性的语句的编写来实现特定的需求&#xff0c;从这里开始需要我们有一个宏观的思想&…

Linux-内核驱动-中断-key

DEV_NAME&#xff1a;设备名称。 wg&#xff1a;等待队列头&#xff0c;用于同步。 condition&#xff1a;条件变量&#xff0c;用于等待和唤醒。 中断处理函数 irq_handler&#xff1a;处理中断请求&#xff0c;设置条件变量并唤醒等待队列中的进程。 文件操作函数 open…

asm汇编源代码之按键处理相关函数

提供5个子程序: 1. 发送按键 sendkey 2. 检测是否有按键 testkey 3. 读取按键 getkey 4. 判断键盘缓冲区是否为空 bufempty 5. 判断键盘缓冲区是否已满 buffull 具体功能及参数描述如下 sendkey proc  far ; axcharcode testkey proc  far ; out: ;   zf1 buff empt…

Java Collections 类中常用方法使用

一、Collections类 java.util.Collections 类是 Java 集合框架中的一个工具类&#xff0c;提供了一系列静态方法来操作和处理各种类型的集合。这些方法简化了对集合进行排序、查找、同步控制、创建只读集合等常见操作的过程。 二、常用方法 方法类别‌‌方法签名‌‌功能‌‌需…

Linux网络编程——数据链路层详解,以太网、MAC地址、MTU、ARP、DNS、NAT、代理服务器......

目录 一、前言 二、以太网 二、以太网帧格式 三、 MAC地址 四、MTU 1、数据链路层的数据分片 2、MTU对UDP协议的影响 3、MTU对TCP协议的影响 五、ARP协议 1、什么是ARP 2、ARP的作用 3、ARP协议的工作流程 4、ARP缓存表 5、ARP请求报文 6、中间人 六、DNS&…

轻量级开源文件共享系统PicoShare本地部署并实现公网环境文件共享

&#xfeff;## 前言 本篇文章介绍&#xff0c;如何在 Linux 系统本地部署轻量级文件共享系统 PicoShare&#xff0c;并结合 Cpolar 内网穿透实现公网环境远程传输文件至本地局域网内文件共享系统。 PicoShare 是一个由 Go 开发的轻量级开源共享文件系统&#xff0c;它没有文…