Class文件结构

目录

基本概念

字节码的解释

常量池


基本概念

1.编译成字节码文件(.class)并不是只能Java能整

2.Class文件包含了三大类,两大数据结构

三大类:虚拟机指令,符号表,其他辅助信息

两大数据结构:无符号数,表(_info,类似一个结构)

3.计算机以010101010方式来执行。

4.类文件字段类型:

u1表示一个字节,u2表示二个字节,u4表示四个字节,u8表示八个字节

_info类型是表数据结构,对照相应的表进行分析,例如:常量池(cp_info)的分析,就对应着Class文件结构-常量池表

 

字节码的解释

常量池

类的元数据:描述类的数据。

1.魔数和版本

 

2.常量池计数器:两个字节表示(00  16)

十六进制转十进制:例如 16 = 6*16^0 + 1* 16^1 = 6 + 16 = 22

也就是有22个常量,但是实际上只有21个.留了一个空位

3.对照常量池表分析Class文件结构

(1)使用javap -v 命令(反汇编命令),对照常量池结构表分析对应的Class文件的常量即可;

(2)简单的字节码指令

(3)复杂一点的字节码

--解读虚拟机指令(invokespecial)

public class jvm.classFile.Test1
  minor version: 0
  major version: 52
  flags: ACC_PUBLIC, ACC_SUPER
Constant pool:
   #1 = Methodref          #3.#13         // java/lang/Object."<init>":()V
   #2 = Class              #14            // jvm/classFile/Test1
   #3 = Class              #15            // java/lang/Object
   #4 = Utf8               <init>
   #5 = Utf8               ()V
   #6 = Utf8               Code
   #7 = Utf8               LineNumberTable
   #8 = Utf8               LocalVariableTable
   #9 = Utf8               this
  #10 = Utf8               Ljvm/classFile/Test1;
  #11 = Utf8               SourceFile
  #12 = Utf8               Test1.java
  #13 = NameAndType        #4:#5          // "<init>":()V
  #14 = Utf8               jvm/classFile/Test1
  #15 = Utf8               java/lang/Object
{
  public jvm.classFile.Test1();
    descriptor: ()V
    flags: ACC_PUBLIC
    Code:
      stack=1, locals=1, args_size=1
         0: aload_0
         1: invokespecial #1                  // Method java/lang/Object."<init>":()V
         4: return
      LineNumberTable:
        line 7: 0
      LocalVariableTable:
        Start  Length  Slot  Name   Signature
            0       5     0  this   Ljvm/classFile/Test1;
}
SourceFile: "Test1.java"

复杂的

public class jvm.classFile.Test2 extends jvm.classFile.Test1 implements java.io.Serializable,java.lang.Comparable
  minor version: 0
  major version: 52
  flags: ACC_PUBLIC, ACC_SUPER
Constant pool:
   #1 = Methodref          #3.#24         // jvm/classFile/Test1."<init>":()V
   #2 = Class              #25            // jvm/classFile/Test2
   #3 = Class              #26            // jvm/classFile/Test1
   #4 = Class              #27            // java/io/Serializable
   #5 = Class              #28            // java/lang/Comparable
   #6 = Utf8               name
   #7 = Utf8               Ljava/lang/String;
   #8 = Utf8               age
   #9 = Utf8               I
  #10 = Utf8               <init>
  #11 = Utf8               ()V
  #12 = Utf8               Code
  #13 = Utf8               LineNumberTable
  #14 = Utf8               LocalVariableTable
  #15 = Utf8               this
  #16 = Utf8               Ljvm/classFile/Test2;
  #17 = Utf8               mathod
  #18 = Utf8               compareTo
  #19 = Utf8               (Ljava/lang/Object;)I
  #20 = Utf8               o
  #21 = Utf8               Ljava/lang/Object;
  #22 = Utf8               SourceFile
  #23 = Utf8               Test2.java
  #24 = NameAndType        #10:#11        // "<init>":()V
  #25 = Utf8               jvm/classFile/Test2
  #26 = Utf8               jvm/classFile/Test1
  #27 = Utf8               java/io/Serializable
  #28 = Utf8               java/lang/Comparable
{
  public jvm.classFile.Test2();
    descriptor: ()V
    flags: ACC_PUBLIC
    Code:
      stack=1, locals=1, args_size=1
         0: aload_0
         1: invokespecial #1                  // Method jvm/classFile/Test1."<init>":()V
         4: return
      LineNumberTable:
        line 10: 0
      LocalVariableTable:
        Start  Length  Slot  Name   Signature
            0       5     0  this   Ljvm/classFile/Test2;

  public void mathod();
    descriptor: ()V
    flags: ACC_PUBLIC
    Code:
      stack=0, locals=1, args_size=1
         0: return
      LineNumberTable:
        line 18: 0
      LocalVariableTable:
        Start  Length  Slot  Name   Signature
            0       1     0  this   Ljvm/classFile/Test2;

  public int compareTo(java.lang.Object);
    descriptor: (Ljava/lang/Object;)I
    flags: ACC_PUBLIC
    Code:
      stack=1, locals=2, args_size=2
         0: iconst_0
         1: ireturn
      LineNumberTable:
        line 23: 0
      LocalVariableTable:
        Start  Length  Slot  Name   Signature
            0       2     0  this   Ljvm/classFile/Test2;
            0       2     1     o   Ljava/lang/Object;
}
SourceFile: "Test2.java"

解释:

第一步,到常量池里面找#1常量(#1:MethodRef #3,#24)

        #3 指向Class_info的索引项(u2类型数据)

        #24 指向的就NameAndType的索引项(u2类型数据)

第二步:找#3号常量(#26)

第三步:找#26号常量(#26号常量是utf-8编码的字符串: Test1)

第四步:找#24号常量,NameAndType的数据结构(u2:名称常量项的索引;u2:描述符常量项的索引;#10:#11)

第五步:#10对应<init>

第六步:#11对应()V

搜索的结果:V  Test1.<init>()

 

4.init和clinit

init:实例化初始化方法

调用new初始化对象的时候;

调用反射的时候newInstance();

调用clone方法的时候

ObjectInpustream.getObject序列化的时候;

clinit:类和接口的初始化

所有的类变量初始化语句和静态初始化语句都被Java编译搜集到一起,放到clinit;

 

访问标识

用于识别一些类或者接口层次的访问信息,包括:这个Class是类还是接口,是否为public,是否为abstract,类是否声明为final等

访问标识之后顺序排列:类索引,父类索引,接口索引集合;

接口索引集合入口第一项是u2类型的接口计数器,表示索引表的容量(实现的接口书),如果该类没有实现任何接口,则计数器的值为0,实现多少个接口,每个接口都需要u2个单位

 

 

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值