JVM Knowleadge-字节码文件的构造

2024-04-09 20:08

本文主要是介绍JVM Knowleadge-字节码文件的构造,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

本文参考了周志明的《深入理解Java虚拟机:JVM高级特性与最佳实践》,这本书写的非常好!极力推荐!

字节码平台无关性:

Java的一次编译到处运行即要求所载入的和执行同一种平台无关的字节码。各个平台的虚拟机与所有平台都统一采用字节码(ByteCode)存储格式。


有很多语言可以在JVM上执行。而语言无关性的基础是:JVM和字节码存储格式。

JVM不关心Class的来源是什么语言,之后要它符合Class文件应有的结构就可以在JVM上执行。


使用命令获取的一个字节码阅读文件:

使用Javap-verbose xxx.class命令得到的一个结果



字节码文件组成:


构成细节如下:

Java语言中的各种变量,关键字和运算符号的语义最终都是由多条字节码命令组合而成的。

  1. Class文件是以8位字节为基础单位的二进制流。

    1. 各个数据项按照顺序紧密排列在Class文件之中,中间没有添加任何分隔符。

    2. 8位以上的数据项会分割成若干个8位字节进行存储。

  2. Class文件格式采用类似于C语言结构体的伪结构来存储。这种伪结构只有两种类型的数据类型:无符号数和表。

    1. 无符号数

      1. 属于基本数据类型,以u1,u2,u4,u8来分别代表1个字节,2个字节,4个字节,8个字节的无符号数。

      2. 用来描述数字,索引引用,数量值,或者UTF-8编码的字符串值。

      1. 由多个无符号数或者其他表作为数据项构成的符合数据类型。

      2. 习惯用_info结尾。

      3. 用来描述有层次关系的复合结构的数据。整个Class文件实质上就是一张表。

    2. 无论无符号数还是表,当需要描述同一类型但数量不定的多个数据时,经常会使用一个前置的容量计数器+若干个连续的数据项形式。

  1. Class文件构成细节

  1. MagicNumber:每个Class文件的u4称为MagicNumber,它的唯一作用是用来确定文件类型。

  1. Class文件的版本号:紧跟着MagicNumberu4Class文件的版本编号。

  2. 常量池

    1. 版本号后面的是常量池的入口。

    2. 常量池与其他项目关联最多,占用空间较大。

    3. 常量池的数量通常是不固定的。所以常量池的入口需要放置一个u2类型的数据,代表常量池的容量计数器。

    4. 存储两类常量:

      1. 字面量(Literal):文本字符串,final的常量值等。

      2. 符号引用(SymbolicReferences)

        1. 类和接口的全限定名。

        2. 字段的名称和描述符。

        3. 方法的名称和描述符。

注意:在Class文件中不会保存各个方法和字段的最终内存布局,而是符号引用。


    1. 常量池中每一项常量都是一个表,共有11种不同的的表结构。

      1. CONSTANT_Utf8_info(utf-8编码的字符串)

      2. 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(字段或方法的部分符号引用)

    2. 11种表中开始的第一位是一个u1类型的标志位,代表当前这个常量属于那种类型的。

    3. Class文件中方法,字段都需要引用CONSTANT_Utf8_info常量来表示名称。

    4. 使用Javap-verbose xxx.class命令可以字节码内容。


  1. 常量池结束以后,紧跟着2个字节的访问标志(access_flags),这个标志用来识别一些类或接口层次的访问信息。

    1. ACC_PUBLIC:是否为public类型

    2. ACC_FINAL:是否被声明为final

    3. ACC_SUPERJDK1.2之后,这个值为真


    1. ACC_INTERFACE:标识这是一个接口

    2. ACC_ABSTRACT:是否为abstract,对于抽象类或接口来说,此值为真


    1. ACC_SYNTHETIC:标识这个类并非由用户代码产生。

    2. ACC_ANNOTION:标识这是一个注解

    3. ACC_ENUM:标识这是一个枚举

  1. 类索引(this_class),父类索引(super_class),接口索引(interfaces)集合

    1. 这些数据项位于access_flags之后。

    2. Class文件由这三项数据来确定这个类的继承关系。

    3. 类索引用于确定这个类的全限定名,父类索引用于确定这个类的父类的全限定名。

    4. Interfaces是一个集合,因为一个类可实现多个接口。

    5. 除了java.lang.Object之外,所有的java类的父类索引都不为0

    6. 对于索引集合,入口的第一项为该索引表的容量(代表实现接口的数量)。

  2. 字段表(filed_info)

    1. 用于描述接口类或类中声明的变量。

    2. Filed包括static和非static的。

    3. Filed不包括局部变量。

    4. Field信息:

      1. 作用域(private,protected,default,public)

      2. StaicorStatic

      3. 可变性final

      4. 并发可见性(volatile)

      5. 可否序列化(transient)

      6. 字段的数据类型(基本类型,对象,数组),字段名称。


这篇关于JVM Knowleadge-字节码文件的构造的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/889088

相关文章

JVM 的类初始化机制

前言 当你在 Java 程序中new对象时,有没有考虑过 JVM 是如何把静态的字节码(byte code)转化为运行时对象的呢,这个问题看似简单,但清楚的同学相信也不会太多,这篇文章首先介绍 JVM 类初始化的机制,然后给出几个易出错的实例来分析,帮助大家更好理解这个知识点。 JVM 将字节码转化为运行时对象分为三个阶段,分别是:loading 、Linking、initialization

Spring Security 基于表达式的权限控制

前言 spring security 3.0已经可以使用spring el表达式来控制授权,允许在表达式中使用复杂的布尔逻辑来控制访问的权限。 常见的表达式 Spring Security可用表达式对象的基类是SecurityExpressionRoot。 表达式描述hasRole([role])用户拥有制定的角色时返回true (Spring security默认会带有ROLE_前缀),去

浅析Spring Security认证过程

类图 为了方便理解Spring Security认证流程,特意画了如下的类图,包含相关的核心认证类 概述 核心验证器 AuthenticationManager 该对象提供了认证方法的入口,接收一个Authentiaton对象作为参数; public interface AuthenticationManager {Authentication authenticate(Authenti

Spring Security--Architecture Overview

1 核心组件 这一节主要介绍一些在Spring Security中常见且核心的Java类,它们之间的依赖,构建起了整个框架。想要理解整个架构,最起码得对这些类眼熟。 1.1 SecurityContextHolder SecurityContextHolder用于存储安全上下文(security context)的信息。当前操作的用户是谁,该用户是否已经被认证,他拥有哪些角色权限…这些都被保

Spring Security基于数据库验证流程详解

Spring Security 校验流程图 相关解释说明(认真看哦) AbstractAuthenticationProcessingFilter 抽象类 /*** 调用 #requiresAuthentication(HttpServletRequest, HttpServletResponse) 决定是否需要进行验证操作。* 如果需要验证,则会调用 #attemptAuthentica

Spring Security 从入门到进阶系列教程

Spring Security 入门系列 《保护 Web 应用的安全》 《Spring-Security-入门(一):登录与退出》 《Spring-Security-入门(二):基于数据库验证》 《Spring-Security-入门(三):密码加密》 《Spring-Security-入门(四):自定义-Filter》 《Spring-Security-入门(五):在 Sprin

Java架构师知识体认识

源码分析 常用设计模式 Proxy代理模式Factory工厂模式Singleton单例模式Delegate委派模式Strategy策略模式Prototype原型模式Template模板模式 Spring5 beans 接口实例化代理Bean操作 Context Ioc容器设计原理及高级特性Aop设计原理Factorybean与Beanfactory Transaction 声明式事物

Java进阶13讲__第12讲_1/2

多线程、线程池 1.  线程概念 1.1  什么是线程 1.2  线程的好处 2.   创建线程的三种方式 注意事项 2.1  继承Thread类 2.1.1 认识  2.1.2  编码实现  package cn.hdc.oop10.Thread;import org.slf4j.Logger;import org.slf4j.LoggerFactory

字节面试 | 如何测试RocketMQ、RocketMQ?

字节面试:RocketMQ是怎么测试的呢? 答: 首先保证消息的消费正确、设计逆向用例,在验证消息内容为空等情况时的消费正确性; 推送大批量MQ,通过Admin控制台查看MQ消费的情况,是否出现消费假死、TPS是否正常等等问题。(上述都是临场发挥,但是RocketMQ真正的测试点,还真的需要探讨) 01 先了解RocketMQ 作为测试也是要简单了解RocketMQ。简单来说,就是一个分

JAVA智听未来一站式有声阅读平台听书系统小程序源码

智听未来,一站式有声阅读平台听书系统 🌟 开篇:遇见未来,从“智听”开始 在这个快节奏的时代,你是否渴望在忙碌的间隙,找到一片属于自己的宁静角落?是否梦想着能随时随地,沉浸在知识的海洋,或是故事的奇幻世界里?今天,就让我带你一起探索“智听未来”——这一站式有声阅读平台听书系统,它正悄悄改变着我们的阅读方式,让未来触手可及! 📚 第一站:海量资源,应有尽有 走进“智听