Java 虚拟机原理Class 字节码二进制文件分析 五 ( 方法计数器 | 方法表 | 访问标志 | 方法名称索引 | 方法返回值类型 | 方法属性数量 | 方法属性表 )

Posted 韩曙亮

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了Java 虚拟机原理Class 字节码二进制文件分析 五 ( 方法计数器 | 方法表 | 访问标志 | 方法名称索引 | 方法返回值类型 | 方法属性数量 | 方法属性表 )相关的知识,希望对你有一定的参考价值。

前言

上一篇博客 【Java 虚拟机原理】Class 字节码二进制文件分析 四 ( 字段表数据结构 | 字段表详细分析 | 访问标志 | 字段名称 | 字段描述符 | 属性项目 ) 分析了字段表的一些数据 ;

当前的字节码文件中只有 1 1 1 个字段 , 字段表显示 1 1 1 个字段结束后 , 后面的字节是 方法计数器 和 方法表 的数据 ;

本篇博客中 , 继续向后分析 字节码对应数据 ;

分析的原始数据是 【Java 虚拟机原理】Class 字节码二进制文件分析 一 ( 字节码文件附加信息 | 魔数 | 次版本号 | 主版本号 | 常量池个数 ) 二、字节码文件示例 章节中的 Java 源码 , Class 字节码 , 字节码附加信息 ;

public class Student {
    private String name;

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }
}

字节码附加信息 :

D:\\jvm>javap -v Student.class
Classfile /D:/jvm/Student.class
  Last modified 2021-9-4; size 392 bytes
  MD5 checksum 8b9bb897bb8cf2a8addf04be5b7b915f
  Compiled from "Student.java"
public class Student
  minor version: 0
  major version: 52
  flags: ACC_PUBLIC, ACC_SUPER
Constant pool:
   #1 = Methodref          #4.#17         // java/lang/Object."<init>":()V
   #2 = Fieldref           #3.#18         // Student.name:Ljava/lang/String;
   #3 = Class              #19            // Student
   #4 = Class              #20            // java/lang/Object
   #5 = Utf8               name
   #6 = Utf8               Ljava/lang/String;
   #7 = Utf8               <init>
   #8 = Utf8               ()V
   #9 = Utf8               Code
  #10 = Utf8               LineNumberTable
  #11 = Utf8               getName
  #12 = Utf8               ()Ljava/lang/String;
  #13 = Utf8               setName
  #14 = Utf8               (Ljava/lang/String;)V
  #15 = Utf8               SourceFile
  #16 = Utf8               Student.java
  #17 = NameAndType        #7:#8          // "<init>":()V
  #18 = NameAndType        #5:#6          // name:Ljava/lang/String;
  #19 = Utf8               Student
  #20 = Utf8               java/lang/Object
{
  public Student();
    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 1: 0

  public java.lang.String getName();
    descriptor: ()Ljava/lang/String;
    flags: ACC_PUBLIC
    Code:
      stack=1, locals=1, args_size=1
         0: aload_0
         1: getfield      #2                  // Field name:Ljava/lang/String;
         4: areturn
      LineNumberTable:
        line 5: 0

  public void setName(java.lang.String);
    descriptor: (Ljava/lang/String;)V
    flags: ACC_PUBLIC
    Code:
      stack=2, locals=2, args_size=2
         0: aload_0
         1: aload_1
         2: putfield      #2                  // Field name:Ljava/lang/String;
         5: return
      LineNumberTable:
        line 9: 0
        line 10: 5
}
SourceFile: "Student.java"




一、方法表结构



字段表之后的 2 2 2 字节就是方法计数器 , 记录当前字节码有多少个方法 ;

方法计数器 后 就是方法表 , 方发表中每个方法都是 一个 method_info 数据结构 ;

假如 方法计数器 中 方法个数是 3 3 3 , 那么方法表中有 3 3 3 个 method_info ;

方法表 method_info 结构 : 方发表结构 与 字段表结构一样 ;

方发表就是若干上述 method_info 结构的排列 , 方法表的方法排列完毕后 , 是属性计数器 和 属性表 ;





二、方法计数器



方法计数器的值为 00 03 , 说明有 3 3 3 个方法 ;





三、方法表数据解析 ( init 构造方法 )



这是字节码 附加信息 中的构造方法数据 :

  public Student();
    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 1: 0

1、方法访问标志


access_flag ( 方法访问权限 ) : 方法计数器 后 的 2 2 2 字节 , 是 方发表中 第一个 method_info 方法信息的 方法访问标志 ;

00 01 表示该方法是 public 的 ;

方法访问标志 :


2、方法名称索引


name_index ( 方法名称索引 ) : 2 2 2 字节 , 00 07 表示常量表中的 #7 , 常量为 :

Constant pool:
   #7 = Utf8               <init>

<init> 方法指的是构造方法 ;


3、方法返回类型


descriptor_index ( 方法返回类型 ) : 2 2 2 字节 , 00 08 表示常量表中的 #8 , 常量为 :

Constant pool:
   #8 = Utf8               ()V

()V 方法的返回类型是 void , 没有返回类型 ;


4、方法属性数量


attributes_count ( 方法属性数量 ) : 2 2 2 字节 , 00 01 表示方法属性数量 1 1 1

如果该方法的属性数量不为 0 0 0 , 则之后的数据就是属性表数据 ;

下一篇博客开始分析属性表 ;

以上是关于Java 虚拟机原理Class 字节码二进制文件分析 五 ( 方法计数器 | 方法表 | 访问标志 | 方法名称索引 | 方法返回值类型 | 方法属性数量 | 方法属性表 )的主要内容,如果未能解决你的问题,请参考以下文章

Java 虚拟机原理Class 字节码二进制文件分析 四 ( 字段表数据结构 | 字段表详细分析 | 访问标志 | 字段名称 | 字段描述符 | 属性项目 )

Java 虚拟机原理Class 字节码二进制文件分析 三 ( 访问和修饰标志 | 类索引 | 父类索引 | 接口计数器 | 接口表 | 字段计数器 | 字段表 )

Java 虚拟机原理Class 字节码二进制文件分析 六 ( 属性类型 | Code 属性 | 属性名称索引 | 属性长度 | 操作数栈最大深度 | 局部变量存储空间 | 字节码长度 )

Java 虚拟机原理动态字节码技术 | Dalvik & ART 虚拟机 | Android 字节码打包过程

Java 虚拟机原理Class 字节码二进制文件分析 五 ( 方法计数器 | 方法表 | 访问标志 | 方法名称索引 | 方法返回值类型 | 方法属性数量 | 方法属性表 )

Java 虚拟机原理Class 字节码二进制文件分析 二 ( 常量池位置 | 常量池结构 | tag | info[] | 完整分析字节码文件中的常量池二进制数据 )