JVM调优篇之JVM基础入门AND字节码文件解读
目录
- Java程序编译
- class文件内容
- 常量池
- 附录-访问标识表
- 附录-常量池类型列表
Java程序编译
Java文件通过编译成class文件后,通过JVM虚拟机解释字节码文件转为操作系统执行的二进制码运行。

规范
Java虚拟机有自己的一套规范,遵循这套规范,任何形式的语言都可以在JVM上运行,前提是编译成class文件,此规范相见oracle官网文档。
实现
JVM虚拟机落地实现有很多,常见的有:
- Hotspot,使用最热门的JVM
- OpenJ9,IBM开发
- MicroSoft JVM 微软开发
- TaobaoVM 淘宝开发
- azul zing VM 垃圾回收行业标杆,最快
class文件内容
从上图内容可看出class文件包以下几个方面内容:
- 魔数(
magic number)占4个字节,固定的值0xcafebabe,主要来校验class文件。 - version 版本号,分为小版本号
minor version和大版本号major version,各占两个字节。 - 常量池信息(
constant_pool),具体的各种常量信息,字符串,类名,方法名,字段名等。 - 访问标识(
access_flag),2个字节描述类与接口的修饰符,有几个常见的常量值见附录-访问标识 - 当前类(
this_class) - 当前类的父类(
super_class) - 具体的接口信息(
interfaces) - 具体的属性信息(
fileds) - 具体的方法信息(
methods) - 其它附加属性信息(
attributes)
更具体的内容如下:
| 项目 | Value |
|---|---|
| Magic Number | 魔数(CAFE BABE) |
| Minor Number | 次版本号 |
| Major Number | 主版本号 |
| constant_pool_count | 常量池数量 |
| constant_pool | 常量池具体实现,是一种表 |
| access_flag | 访问标识 |
| this_class | 当前类名 |
| super_class | 父类名称 |
| interfaces_count | 接口数量 |
| interfaces | 具体实现的接口信息 |
| fileds_count | 属性数量 |
| fileds | 具体属性信息 |
| methods_count | 方法数量 |
| methods | 具体方法的信息 |
| attributes_count | 其它附加属性的数量 |
| attributes | 其它附加属性具体信息 |
现在来看具体示例来看上面的代表是什么意思,准备一个简单的类:
public class ByteCode01 {public ByteCode01() {}
}
编译之后的字节码文件:

class文件中就是一个二进制字节流,非0即1,数据类型有五种,u1,u2,u4,u8和_info表类型。
上述这种文件需要借助工具来分析具体内容,在idea安装jclasslib插件看Java文件编译成class文件信息,选中编译后的class文件,如下图所示:

访问标识是固定的值,通过位运算得出的0x0021既代表public又代表super(0x0021是0x0001与0x0020位与运算),详见附录-访问标识表
更直观的表现:

常量池
本类索引即当前类名(this_class),cp_info是constant_pool的表类型,#2是第二张表,cp_info #2代表常量池第二张表,存的是对cp_info #14的引用,cp_info #14存的是具体的字符串值,就是当前类的类名。


class文件中都有属于自己的常量池,常量池包含类型列表,来存储类、方法、字段、接口等信息,以及各种字面量,通过解析常量池可以访问这些字符串信息。比如上图出现的CONSTANT_Utf8_info、CONSTANT_Class_info等详见文末常量池类型列表。
着重看下CONSTANT_Utf8_info,它用来存储 UTF-8 编码的字符串,相当于一张常量池表,看一下它的具体结构:
| 字段名 | 字节数 | 描述 |
|---|---|---|
| tag | 1 | CONSTANT_Utf8_info的常量类型是1(其余的见文末常量池类型列表) |
| length | 2 | u2(2 字节无符号数) 类型字段,表示 bytes 部分字节长度,以字节为单位的 |
| bytes | length | 实际字符串数据,采用 UTF-8 编码显示 |
比如
public class Test {public static void main(String[] args) {System.out.println("Hello, World!");}
}
字符串 Hello, World!,会以 CONSTANT_Utf8_info 类型存储在常量池中,对应的二进制内容或许是:
tag = 0x01 (标识 CONSTANT_Utf8_info)
length = 0x000D (字符串的字节长度为 13)
bytes = 48 65 6C 6C 6F 2C 20 57 6F 72 6C 64 21 ("Hello, World!" 的 UTF-8 编码)
常量池通常分为下面几种常量值类型:
| 类型 | 标志(tag) | 描述 |
|---|---|---|
| CONSTANT_Utf8_info | 1 | UTF-8编码的字符串 |
| CONSTANT_Integer_info | 3 | 整型字面量 |
| CONSTANT_Float_info | 4 | 浮点型字面 |
| CONSTANT_Long_info | 5 | 长整型字面量 |
| CONSTANT_Double_info | 6 | 双精度浮点型字面量 |
| CONSTANT_Class_info | 7 | 类或接口的符号引用 |
| CONSTANT_String_info | 8 | 字符串类型字面量 |
| CONSTANT_Fieldref_info | 9 | 字段的符号引用 |
| CONSTANT_Methodref_info | 10 | 类中方法的符号引用 |
| CONSTANT_InterfaceMethodref_info | 11 | 接口中方法的符号引用 |
| CONSTANT_NameAndType_info | 12 | 字段或方法的部分符号引用 |
| CONSTANT_MethodHandle_info | 15 | 表示方法句柄 |
| CONSTANT_MethodType_info | 16 | 表示方法类型 |
| CONSTANT_InvokeDynamic_info | 18 | 表示一个动态方法调用点 |
附录-访问标识表
| 项目 | Value |
|---|---|
| ACC_PUBLIC 0x0001 | 是否是public类型 |
| ACC_FINAL 0x0010 | 是否是NULL |
| ACC_SUPER 0x0020 | 该标志必须为真 |
| ACC_INTERFACE 0x0200 | 是否为接口 |
| ACC_ABSTRACT 0X0400 | 是否是接口或者抽象类 |
| ACC_SYNTHETIC 0x1000 | 编译器自动生成,非用户代码产生 |
| ACC_ANNOTATION 0x2000 | 是否为注解 |
| ACC_ENUM 0x4000 | 是否为枚举 |
附录-常量池类型列表
| 常量类型 | tag值 | 描述 |
|---|---|---|
| CONSTANT_Utf8_info | 1 | 用于存储 UTF-8 编码的字符串 |
| CONSTANT_Integer_info | 3 | 用于存储 4 字节的整型字面量(int) |
| CONSTANT_Float_info | 4 | 用于存储 4 字节的浮点型字面量(float) |
| CONSTANT_Long_info | 5 | 用于存储 8 字节的长整型字面量(long) |
| CONSTANT_Double_info | 6 | 用于存储 8 字节的双精度浮点型字面量(double) |
| CONSTANT_Class_info | 7 | 用于存储类或接口的符号引用,引用 CONSTANT_Utf8_info 中的类名 |
| CONSTANT_String_info | 8 | 用于存储字符串字面量,引用 CONSTANT_Utf8_info 中的字符串内容 |
| CONSTANT_Fieldref_info | 9 | 用于存储字段的符号引用,引用类和字段的描述信息 |
| CONSTANT_Methodref_info | 10 | 用于存储类中方法的符号引用,引用类和方法的描述信息 |
| CONSTANT_InterfaceMethodref_info | 11 | 用于存储接口中方法的符号引用,引用接口和方法的描述信息 |
| CONSTANT_NameAndType_info | 12 | 用于描述字段或方法的名称和类型,引用名称(Utf8)和描述符(Utf8) |
| CONSTANT_MethodHandle_info | 15 | 用于存储对方法句柄的引用(invokedynamic 指令用) |
| CONSTANT_MethodType_info | 16 | 用于存储方法类型的符号引用,指向方法描述符(Utf8) |
| CONSTANT_Dynamic_info | 17 | 用于存储动态调用点的信息(JDK 11 引入,支持动态常量) |
| CONSTANT_InvokeDynamic_info | 18 | 用于存储动态方法调用点的信息,引用引导方法和动态调用名称 |
| CONSTANT_Module_info | 19 | 用于存储模块的符号引用(JDK 9 引入) |
| CONSTANT_Package_info | 20 | 用于存储包的符号引用(JDK 9 引入) |
相关文章:
JVM调优篇之JVM基础入门AND字节码文件解读
目录 Java程序编译class文件内容常量池附录-访问标识表附录-常量池类型列表 Java程序编译 Java文件通过编译成class文件后,通过JVM虚拟机解释字节码文件转为操作系统执行的二进制码运行。 规范 Java虚拟机有自己的一套规范,遵循这套规范,任…...
EXCEL截取某一列从第一个字符开始到特定字符结束的字符串到新的一列
使用EXCEL中的公式进行特定截取 假设列A是一组产品的编码,我们需要的数据是“-”之前的字段。 我们需要在B1单元格输入公式“LEFT(A1,SEARCH("-",A1)-1)”然后选中B1至B4单元格,按“CTRLD”向下填充,就可以得出其它几行“-”之前的…...
数据库期末复习题库
1. Mysql日志功能有哪些? 记录日常操作和错误信息,以便了解Mysql数据库的运行情况,日常操作,错误信息和进行相关的优化。 2. 数据库有哪些备份方法 完全备份:全部都备份一遍表备份:只提取数据库中的数据࿰…...
私有库gitea安装
一 gitea是什么 Gitea是一款自助Git服务,简单来说,就是可以一个私有的github。 搭建很容易。 Gitea依赖于Git。 类似Gitea的还有GitHub、Gitee、GitLab等。 以下是安装步骤。 二 安装sqilite 参考: 在windows上安装sqlite 三 安装git…...
关于最近win11不能使用ie,而不能使用考试客户端的解决方法
弄ie的那个我感觉是非常难的,所以我的是另一种的方法 下载360浏览器(不是360全家桶)360安全浏览器-全面保护上网安全,4亿用户共同选择(上面的是官网,不要下载错了,还有安装界面注意不要勾选一下…...
深度学习之Mask-R-CNN
1.1 Mask-RCNN 的网络结构示意图 其中黑色部分为原来的Faster-RCNN,红色部分为在Faster网络上的修改: 1)将ROI Pooling层替换成了ROIAlign; 2)添加并列的FCN层(Mask层); …...
css包含块
包含块 出现 在css中一些属性的计算可能超出你的预料,在普遍情况下会认为定位属性和百分比的宽高是根据父元素计算的,但是准确来说他们都是根据元素所在的包含块来计算的,所以掌握包含块的知识是非常关键的。 内容 在CSS中,“…...
混沌工程/混沌测试/云原生测试/云平台测试
背景 私有云/公有云/混合云等具有复杂,分布式,环境多样性等特点,许多特殊场景引发的线上问题很难被有效发现。所以需要引入混沌工程,建立对系统抵御生产环境中失控条件的能力以及信心,提高系统面对未知风险得能力。 …...
研发设计数字化:PLM、PDM、ERP介绍及其区别
一、产品全生命周期管理的定义 1.1 产品全生命周期(PLM)发展背景 目前,数字化设计与制造的技术(如CAX、DFX等)已经在产品开发中得到广泛应用,而各种企业和产品管理软件(如ERP、SCM、PDM、CRM等…...
Python练习51
Python日常练习 题目: 调用函数fun判断一个三位数是否“水仙花数”。 在main函数中从键盘输入一个三位数,并输 出判断结果。请编写fun函数。 说明: 所谓“水仙花数”是指一3位数,其各位数字立方和 等于该数本…...
Qt 前置课程 QtNFC
文章目录 详解 Qt NFC 模块(QtNFC)1. 什么是 NFC?2. NFC 的原理2.1 主动设备与被动设备2.2 三种工作模式2.3 数据交换 3. QtNFC 模块概述4. 使用 QtNFC 模块4.1 配置 .pro 文件 5. NFC 的常见应用场景6. QtNFC 模块的主要类6.1 QNearFieldMan…...
【论文阅读】 Learning to Upsample by Learning to Sample
论文结构目录 一、之前的上采样器二、DySample概述三、不同上采样器比较四、整体架构五、设计过程(1)初步设计(2)第一次修改(3)第二次修改(4)第三次修改 六、DySample四种变体七、复…...
堆排序(含证明)
引言 前面我们讲过堆的基本操作的实现,现在给定一个int类型的数组,里面存放的数据是无序的,我们如何利用堆的思想来实现数组内数据的升序排列或降序排列呢? 通过前面讲到的堆的实现,我们可以想到,我们再开…...
蓝桥杯模拟题不知名题目
题目:p是一个质数,但p是n的约数。将p称为是n的质因数。求2024最大质因数。 #include<iostream> #include<algorithm> using namespace std; bool fun(int x) {for(int i 2 ; i * i < x ; i){if(x % i 0)return false;}return true; } int main() …...
C#中的工厂模式
在C#中,工厂模式(Factory Pattern) 是一种常见的设计模式,它属于创建型模式,主要用于定义一个用于创建对象的接口,让子类决定实例化哪一个类。通过使用工厂模式,客户端代码不需要直接实例化具体…...
深度学习与持续学习:人工智能的未来与研究方向
文章目录 1. 持续学习与深度学习1.1 深度学习的局限1.2 持续学习的定义 2. 目标与心智2.1 奖励假说2.2 心智的构成 3. 对研究方法的建议3.1 日常写作记录3.2 中立对待流行趋势 1. 持续学习与深度学习 1.1 深度学习的局限 深度学习注重“瞬时学习”,如ChatGPT虽在语…...
OGRE 3D----4. OGRE和QML共享opengl上下文
在现代图形应用开发中,OGRE(Object-Oriented Graphics Rendering Engine)和QML(Qt Modeling Language)都是非常流行的工具。OGRE提供了强大的3D渲染能力,而QML则用于构建灵活的用户界面。在某些应用场景中,我们需要在同一个应用程序中同时使用OGRE和QML,并且共享OpenGL…...
【Umi】常用配置
具体见:alias 1. 基础配置 1)配置别名alias 2)配置sourcemap devtool 配置项 3)添加hash 4)图片转base64 inlineLimit 配置项 5)设置JS压缩方式 jsMinifier (webpack) 、jsMinifierOptions 配置项 6)设置umi插件 plugins 配置项 7)设置打包后资源导入的路…...
Windows加固脚本
echo off REM 清屏 cls title 安全策略设置批处理 color f0 echo **************************************** echo write by afei echo https://www.jianshu.com/u/ea4c85fbe8c7 echo **************************************** pause cls color 3f echo ********************…...
玩游戏常常出现vc++runtime library error R6025 这是什么意思,该怎么解决?
当玩游戏时常常出现“vc runtime library error R6025”错误,这通常表明微软C开发运行库组件存在问题。以下是对该错误及其解决方法的详细解释: 错误含义 “vc runtime library error R6025”是一个与Visual C运行时库相关的错误,该错误表明…...
从WWDC看苹果产品发展的规律
WWDC 是苹果公司一年一度面向全球开发者的盛会,其主题演讲展现了苹果在产品设计、技术路线、用户体验和生态系统构建上的核心理念与演进脉络。我们借助 ChatGPT Deep Research 工具,对过去十年 WWDC 主题演讲内容进行了系统化分析,形成了这份…...
相机Camera日志实例分析之二:相机Camx【专业模式开启直方图拍照】单帧流程日志详解
【关注我,后续持续新增专题博文,谢谢!!!】 上一篇我们讲了: 这一篇我们开始讲: 目录 一、场景操作步骤 二、日志基础关键字分级如下 三、场景日志如下: 一、场景操作步骤 操作步…...
【机器视觉】单目测距——运动结构恢复
ps:图是随便找的,为了凑个封面 前言 在前面对光流法进行进一步改进,希望将2D光流推广至3D场景流时,发现2D转3D过程中存在尺度歧义问题,需要补全摄像头拍摄图像中缺失的深度信息,否则解空间不收敛…...
Java多线程实现之Callable接口深度解析
Java多线程实现之Callable接口深度解析 一、Callable接口概述1.1 接口定义1.2 与Runnable接口的对比1.3 Future接口与FutureTask类 二、Callable接口的基本使用方法2.1 传统方式实现Callable接口2.2 使用Lambda表达式简化Callable实现2.3 使用FutureTask类执行Callable任务 三、…...
【HarmonyOS 5 开发速记】如何获取用户信息(头像/昵称/手机号)
1.获取 authorizationCode: 2.利用 authorizationCode 获取 accessToken:文档中心 3.获取手机:文档中心 4.获取昵称头像:文档中心 首先创建 request 若要获取手机号,scope必填 phone,permissions 必填 …...
Redis:现代应用开发的高效内存数据存储利器
一、Redis的起源与发展 Redis最初由意大利程序员Salvatore Sanfilippo在2009年开发,其初衷是为了满足他自己的一个项目需求,即需要一个高性能的键值存储系统来解决传统数据库在高并发场景下的性能瓶颈。随着项目的开源,Redis凭借其简单易用、…...
【MATLAB代码】基于最大相关熵准则(MCC)的三维鲁棒卡尔曼滤波算法(MCC-KF),附源代码|订阅专栏后可直接查看
文章所述的代码实现了基于最大相关熵准则(MCC)的三维鲁棒卡尔曼滤波算法(MCC-KF),针对传感器观测数据中存在的脉冲型异常噪声问题,通过非线性加权机制提升滤波器的抗干扰能力。代码通过对比传统KF与MCC-KF在含异常值场景下的表现,验证了后者在状态估计鲁棒性方面的显著优…...
【从零开始学习JVM | 第四篇】类加载器和双亲委派机制(高频面试题)
前言: 双亲委派机制对于面试这块来说非常重要,在实际开发中也是经常遇见需要打破双亲委派的需求,今天我们一起来探索一下什么是双亲委派机制,在此之前我们先介绍一下类的加载器。 目录 编辑 前言: 类加载器 1. …...
从面试角度回答Android中ContentProvider启动原理
Android中ContentProvider原理的面试角度解析,分为已启动和未启动两种场景: 一、ContentProvider已启动的情况 1. 核心流程 触发条件:当其他组件(如Activity、Service)通过ContentR…...
《Docker》架构
文章目录 架构模式单机架构应用数据分离架构应用服务器集群架构读写分离/主从分离架构冷热分离架构垂直分库架构微服务架构容器编排架构什么是容器,docker,镜像,k8s 架构模式 单机架构 单机架构其实就是应用服务器和单机服务器都部署在同一…...
