Class类文件结构

Class类文件结构

class文件是一组以8位字节为基础单位的二进制流,各个数据项目严格按照顺序紧凑的排列在class文件之中,中间没有添加任何分隔符,这使得整个Class文件中存储的内容几乎全部是程序运行的必要数据,没有空隙存在。

根据java虚拟机规范的规定,Class文件格式采用一种类似于C语言结构体的伪结构来存储数据,这种伪结构只有两种数据类型:无符号数和表,后面的解析都要以这两种数据类型为基础。

无符号数:属于基本的数据类型,以u1,u2,u4,u8来分别代表一个字节,2个字节等~,可以用来描述数字,索引引用,数量值或者按照UTF-8编码构成字符串值。

表:是由多个无符号数或者其他表作为数据项构成的复合数据类型。表用于描述由层次关系的复合结构的数据,整个Class文件本质上就是一张表。

类型 名称 数量
u4 magic 1
u2 minor_version 1
u2 major_version 1
u2 constant_pool_count 1
cp_info constant_pool constant_pool_count - 1
u2 access_flags 1
u2 this_class 1
u2 super_class 1
u2 interfaces_count 1
u2 interfaces interfaces_count
u2 fields_count 1
field_info fields fields_count
u2 methods_count 1
method_info methods methods_count
u2 attribute_count 1
attribute_info attributes attributes_count
    public class Test {
        private int m;

        private int incr() {
            return m + 1;
        }

    }
    
image

魔术与Class文件的版本号

每个Class文件的头4个字节称为魔数(Magic Number),它的唯一作用是用于确定这个文件是否为一个能被虚拟机接受的Class文件。很多文件存储标准中都是用魔数来进行身份识别。
Class文件的魔数的获得很有“浪漫气息”,值为:0xCAFEBABE。

紧接着魔数的4个字节存储的是Class文件的版本号:第5和第6个字节是次版本号(Minor Version),第7个和第8个字节是主版本号(Major Version)。Java的版本号是从45开始的,JDK1.1之后的每个JDK大版本发布主版本号向上加1(JDK1.0–1.1使用了45.0–45.3的版本号),高版本的JDK能向下兼容以前版本的Class文件,但不能运行以后版本的Class文件,即时文件格式并未发生变化。

常量池

紧接着主次版本号之后的是常量池入口,常量池是Class文件结构中与其他项目关联最多的数据类型,也是占用Class文件空间最大的数据项目之一,同时它还是在Class文件中第一个出现的表类型的数据项目。

由于常量池中常量数量不固定,所以在常量池的入口需要放置一项u2类型的数据,代表常量池容量计数值(constant_pool_count)。这个容量计数是从1而不是0开始的。这样做是为了满足后面某些指向常量池的索引值的数据在特定情况下需要表达“不引用任何一个常量池项目”的意思,这种情况就可以把索引值置为0来表示

常量池中主要存放两大类常量:字面量(Literal)和符号引用(Symbolic References)。
字面量比较接近于Java语言层面的常量概念,如文本字符串、被声明为final的常量值等。
而符号引用则属于编译原理方面的概念,包括下面三类常量:

类和接口的全限定名(Fully Qualified Name)
字段的名称和描述符(Descriptor)
方法的名称和描述符

常量池中每一项常量都是一个表,共有14种结构各不相同的表结构数据,这14种表都有一个共同的特点,就是表开始的第一位是一个u1类型的标志位,代表当前这个常量属于哪种常量类型,14种常量类型所代表的具体含义如下表:

类型 标志 含义
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_InterfaceMethod_info 11 接口中方法的符号引用
CONSTANT_NameAndType_info 12 字段或方法的部分符号引用
CONSTANT_MethodHandle_info 15 表示方法句柄
CONSTANT_MethodType_info 16 标识方法类型
CONSTANT_InvokeDynamic_info 18 表示一个动态方法调用点

14种常量类型各自均有自己的结构。

<table border="1" cellspacing="0" cellpadding="0">
<tbody>
<tr>
<td valign="top"><p>常量</p></td>
<td valign="top"><p>项目</p></td>
<td valign="top"><p>类型</p></td>
<td valign="top"><p>含义</p></td>
</tr>
<tr>
<td valign="top" rowspan="3"><p>CONSTANT_Utf8_info</p></td>
<td valign="top"><p>tag</p></td>
<td valign="top"><p>U1</p></td>
<td valign="top"><p>1</p></td>
</tr>
<tr>
<td valign="top"><p>length</p></td>
<td valign="top"><p>U2</p></td>
<td valign="top"><p>UTF-8编码的字符串的长度</p></td>
</tr>
<tr>
<td valign="top"><p>bytes</p></td>
<td valign="top"><p>U1</p></td>
<td valign="top"><p>长度为length的UTF-8编码的字符串</p></td>
</tr>
<tr>
<td valign="top" rowspan="2">
<p>CONSTANT_Integer_info</p>
</td>
<td valign="top">
<p>Tag</p>
</td>
<td valign="top">
<p>U1</p>
</td>
<td valign="top">
<p>3</p>
</td>
</tr><tr><td valign="top">
<p>bytes</p>
</td>
<td valign="top">
<p>U4</p>
</td>
<td valign="top">
<p>按照高位在前的int值</p>
</td>
</tr><tr><td valign="top" rowspan="2">
<p>CONSTANT_Float_info</p>
</td>
<td valign="top">
<p>tag</p>
</td>
<td valign="top">
<p>U1</p>
</td>
<td valign="top">
<p>4</p>
</td>
</tr><tr><td valign="top">
<p>bytes</p>
</td>
<td valign="top">
<p>U4</p>
</td>
<td valign="top">
<p>按照高位在前的float值</p>
</td>
</tr><tr><td valign="top" rowspan="2">
<p>CONSTANT_Long_info</p>
</td>
<td valign="top">
<p>tag</p>
</td>
<td valign="top">
<p>U1</p>
</td>
<td valign="top">
<p>5</p>
</td>
</tr><tr><td valign="top">
<p>bytes</p>
</td>
<td valign="top">
<p>U8</p>
</td>
<td valign="top">
<p>按照高位在前的long值</p>
</td>
</tr><tr><td valign="top" rowspan="2">
<p>CONSTANT_Double_info</p>
</td>
<td valign="top">
<p>tag</p>
</td>
<td valign="top">
<p>U1</p>
</td>
<td valign="top">
<p>6</p>
</td>
</tr><tr><td valign="top">
<p>bytes</p>
</td>
<td valign="top">
<p>U8</p>
</td>
<td valign="top">
<p>按照高位在前的double值</p>
</td>
</tr><tr><td valign="top" rowspan="2">
<p>CONSTANT_Class_info</p>
</td>
<td valign="top">
<p>tag</p>
</td>
<td valign="top">
<p>U1</p>
</td>
<td valign="top">
<p>7</p>
</td>
</tr><tr><td valign="top">
<p>index</p>
</td>
<td valign="top">
<p>U2</p>
</td>
<td valign="top">
<p>指向全限定名常量项的索引</p>
</td>
</tr><tr><td valign="top" rowspan="2">
<p>CONSTANT_String_info</p>
</td>
<td valign="top">
<p>Tag</p>
</td>
<td valign="top">
<p>U1</p>
</td>
<td valign="top">
<p>8</p>
</td>
</tr><tr><td valign="top">
<p>index</p>
</td>
<td valign="top">
<p>U2</p>
</td>
<td valign="top">
<p>指向字符串字面量的索引</p>
</td>
</tr><tr><td valign="top" rowspan="3">
<p>CONSTANT_Fieldref_info</p>
</td>
<td valign="top">
<p>tag</p>
</td>
<td valign="top">
<p>U1</p>
</td>
<td valign="top">
<p>9</p>
</td>
</tr><tr><td valign="top">
<p>index</p>
</td>
<td valign="top">
<p>U2</p>
</td>
<td valign="top">
<p>指向声明字段的类或接口描述符CONSTANT_Class_info的索引项</p>
</td>
</tr><tr><td valign="top">
<p>index</p>
</td>
<td valign="top">
<p>U2</p>
</td>
<td valign="top">
<p>指向字段描述符CONSTANT_NameAndType_info的索引项</p>
</td>
</tr><tr><td valign="top" rowspan="3">
<p>CONSTANT_Methodref_info</p>
</td>
<td valign="top">
<p>tag</p>
</td>
<td valign="top">
<p>U1</p>
</td>
<td valign="top">
<p>10</p>
</td>
</tr><tr><td valign="top">
<p>index</p>
</td>
<td valign="top">
<p>U2</p>
</td>
<td valign="top">
<p>指向声明方法的类描述符CONSTANT_Class_info的索引项</p>
</td>
</tr><tr><td valign="top">
<p>index</p>
</td>
<td valign="top">
<p>U2</p>
</td>
<td valign="top">
<p>指向名称及类描述符CONSTANT_NameAndType_info的索引项</p>
</td>
</tr><tr><td valign="top" rowspan="3">
<p>CONSTANT_InterfaceMethod_info</p>
</td>
<td valign="top">
<p>tag</p>
</td>
<td valign="top">
<p>U1</p>
</td>
<td valign="top">
<p>11</p>
</td>
</tr><tr><td valign="top">
<p>index</p>
</td>
<td valign="top">
<p>U2</p>
</td>
<td valign="top">
<p>指向声明方法的接口描述符COSNTANT_Class_info的索引项</p>
</td>
</tr><tr><td valign="top">
<p>index</p>
</td>
<td valign="top">
<p>U2</p>
</td>
<td valign="top">
<p>指向名称及类描述符CONSTANT_NameAndType_info的索引项</p>
</td>
</tr><tr><td valign="top" rowspan="3">
<p>CONSTANT_NameAndType_info</p>
</td>
<td valign="top">
<p>tag</p>
</td>
<td valign="top">
<p>U1</p>
</td>
<td valign="top">
<p>12</p>
</td>
</tr><tr><td valign="top">
<p>index</p>
</td>
<td valign="top">
<p>U2</p>
</td>
<td valign="top">
<p>指向该字段或方法名称常量池的索引</p>
</td>
</tr><tr><td valign="top">
<p>index</p>
</td>
<td valign="top">
<p>U2</p>
</td>
<td valign="top">
<p>指向该字段或方法描述符常量池的索引</p>
</td>
</tr><tr><td valign="top" rowspan="3">
<p>CONSTANT_MethodHandle_info</p>
</td>
<td valign="top">
<p>tag</p>
</td>
<td valign="top">
<p>U1</p>
</td>
<td valign="top">
<p>15</p>
</td>
</tr><tr><td valign="top">
<p>reference_kind</p>
</td>
<td valign="top">
<p>U2</p>
</td>
<td valign="top">
<p>值必须在1-9之间,决定了方法句柄的类型,方法句柄累心的值表示方法句柄的字节码行为</p>
</td>
</tr><tr><td valign="top">
<p>reference_ index</p>
</td>
<td valign="top">
<p>U2</p>
</td>
<td valign="top">
<p>值必须是对常量池的有效索引</p>
</td>
</tr><tr><td valign="top" rowspan="2">
<p>CONSTANT_MethodType_info</p>
</td>
<td valign="top">
<p>tag</p>
</td>
<td valign="top">
<p>U1</p>
</td>
<td valign="top">
<p>16</p>
</td>
</tr><tr><td valign="top">
<p>descriptor_index</p>
</td>
<td valign="top">
<p>U2</p>
</td>
<td valign="top">
<p>值必须是对常量池的有效索引,常量池在改索引处的项必须是CONSTANT_Utf8_info结构,表示方法的描述符</p>
</td>
</tr><tr><td valign="top" rowspan="3">
<p>CONSTANT_InvokeDynamic_info</p>
</td>
<td valign="top">
<p>tag</p>
</td>
<td valign="top">
<p>U1</p>
</td>
<td valign="top">
<p>18</p>
</td>
</tr><tr><td valign="top">
<p>bootstrap_method_attrindex</p>
</td>
<td valign="top">
<p>U2</p>
</td>
<td valign="top">
<p>值必须是对当前Class文件中引导方法表的bootstrap_methods[]数组的有效索引</p>
</td>
</tr><tr><td valign="top">
<p>name_and_type_index</p>
</td>
<td valign="top">
<p>U2</p>
</td>
<td valign="top">
<p>值必须是对当前常量池的有效索引,常量池在该索引处的项必须是COSTANT_NameAndType_info结构,表示方法名和方法描述符</p>
</td>
</tr></tbody></table>

类或者接口层次的访问信息

access_flag
在常量池结束之后,紧接着的2个字节代表访问标志(access_flag),这个标志用于识别一些类或接口层次的访问信息,包括:这个Class是类还是接口,是否定义为public类型,abstract类型,如果是类的话,是否声明为final,等等。每种访问信息都由一个十六进制的标志值表示,如果同时具有多种访问信息,则得到的标志值为这几种访问信息的标志值的逻辑或。

<table border="1" cellspacing="0" cellpadding="0"><tbody><tr><td valign="top">
<p><span style="font-size:14px;">标志名称</span></p>
</td>
<td valign="top">
<p><span style="font-size:14px;">标志值</span></p>
</td>
<td valign="top">
<p><span style="font-size:14px;">含义</span></p>
</td>
</tr><tr><td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">ACC_PUBLIC</span></p>
</td>
<td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">0x0001</span></p>
</td>
<td valign="top">
<p><span style="font-size:14px;">是否是</span><span style="font-family:'Tw Cen MT';font-size:14px;">public</span></p>
</td>
</tr><tr><td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">ACC_FINAL</span></p>
</td>
<td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">0x0010</span></p>
</td>
<td valign="top">
<p><span style="font-size:14px;">是否被声明为</span><span style="font-family:'Tw Cen MT';font-size:14px;">final</span><span style="font-size:14px;">,只有类可以设置</span></p>
</td>
</tr><tr><td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">ACC_SUPER</span></p>
</td>
<td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">0x0020</span></p>
</td>
<td valign="top">
<p><span style="font-size:14px;">是否允许使用</span><span style="font-family:'Tw Cen MT';font-size:14px;">invokespecial</span><span style="font-size:14px;">字节码指令的新语义,</span><span style="font-family:'Tw Cen MT';font-size:14px;">JDK1.0.2</span><span style="font-size:14px;">之后编译出来的类的这个标志默认为真</span></p>
</td>
</tr><tr><td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">ACC_INTERFACE</span></p>
</td>
<td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">0x0200</span></p>
</td>
<td valign="top">
<p><span style="font-size:14px;">标识是一个接口</span></p>
</td>
</tr><tr><td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">ACC_ABSTRACT</span></p>
</td>
<td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">0x0400</span></p>
</td>
<td valign="top">
<p><span style="font-size:14px;">是否是</span><span style="font-family:'Tw Cen MT';font-size:14px;">abstract</span><span style="font-size:14px;">,对于接口和抽象类来说为真,其他类都为假</span></p>
</td>
</tr><tr><td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">ACC_SYNITHETIC</span></p>
</td>
<td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">0x1000</span></p>
</td>
<td valign="top">
<p><span style="font-size:14px;">标识这个类并非由用户代码产生</span></p>
</td>
</tr><tr><td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">ACC_ANNOTATION</span></p>
</td>
<td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">0x2000</span></p>
</td>
<td valign="top">
<p><span style="font-size:14px;">标识这是一个注解</span></p>
</td>
</tr><tr><td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">ACC_ENUM</span></p>
</td>
<td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">0x4000</span></p>
</td>
<td valign="top">
<p><span style="font-size:14px;">标识这是一个枚举类</span></p>
</td>
</tr></tbody></table>

类索引、父类索引与接口索引集合

由于Java中是单继承,所以父类索引只有一个;但Java类可以实现多个接口,所以接口索引是一个集合。

字段表集合

字段表用来描述接口或类中声明的变量。字段包括类级变量和实例级变量,但不包括方法内变量。

类型 名称 数量
U2 access_flags 1
U2 name_index 1
U2 descriptor_index 1
U2 attributes_count 1
attribute_info attributes attributes_count

字段标志:

<table border="1" cellspacing="0" cellpadding="0"><tbody><tr><td valign="top">
<p><span style="font-size:14px;">标志名称</span></p>
</td>
<td valign="top">
<p><span style="font-size:14px;">标志值</span></p>
</td>
<td valign="top">
<p><span style="font-size:14px;">含义</span></p>
</td>
</tr><tr><td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">ACC_PUBLIC</span></p>
</td>
<td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">0x0001</span></p>
</td>
<td valign="top">
<p><span style="font-size:14px;">字段是否是</span><span style="font-family:'Tw Cen MT';font-size:14px;">public</span></p>
</td>
</tr><tr><td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">ACC_PRIVATE</span></p>
</td>
<td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">0x0002</span></p>
</td>
<td valign="top">
<p><span style="font-size:14px;">字段是否是</span><span style="font-family:'Tw Cen MT';font-size:14px;">private</span></p>
</td>
</tr><tr><td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">ACC_PROTECTED</span></p>
</td>
<td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">0x0004</span></p>
</td>
<td valign="top">
<p><span style="font-size:14px;">字段是否是</span><span style="font-family:'Tw Cen MT';font-size:14px;">protected</span></p>
</td>
</tr><tr><td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">ACC_STATIC</span></p>
</td>
<td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">0x0008</span></p>
</td>
<td valign="top">
<p><span style="font-size:14px;">字段是否是</span><span style="font-family:'Tw Cen MT';font-size:14px;">static</span></p>
</td>
</tr><tr><td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">ACC_FINAL</span></p>
</td>
<td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">0x0010</span></p>
</td>
<td valign="top">
<p><span style="font-size:14px;">字段是否是</span><span style="font-family:'Tw Cen MT';font-size:14px;">final</span></p>
</td>
</tr><tr><td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">ACC_VOLATILE</span></p>
</td>
<td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">0x0040</span></p>
</td>
<td valign="top">
<p><span style="font-size:14px;">字段是否是</span><span style="font-family:'Tw Cen MT';font-size:14px;">volatile</span></p>
</td>
</tr><tr><td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">ACC_TRANSIENT</span></p>
</td>
<td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">0x0080</span></p>
</td>
<td valign="top">
<p><span style="font-size:14px;">字段是否是</span><span style="font-family:'Tw Cen MT';font-size:14px;">transient</span></p>
</td>
</tr><tr><td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">ACC_SYNTHETIC</span></p>
</td>
<td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">0x1000<span></p>
</td>
<td valign="top">
<p><span style="font-size:14px;">字段是否是由编译器自动产生的</span></p>
</td>
</tr><tr><td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">ACC_ENUM</span></p>
</td>
<td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">0x4000</span></p>
</td>
<td valign="top">
<p><span style="font-size:14px;">字段是否是</span><span style="font-family:'Tw Cen MT';font-size:14px;">enum</span></p>
</td>
</tr></tbody></table>

字段描述类型

标识字符 含义
B byte
J long
C char
S short
D double
Z boolean
F float
V void
I int
L 对象类型,如Ljava/lang/Object

字段表集合中不会列出从父类或接口中继承来的字段,但有可能会出现原本Java程序中没有的字段。比较典型的例子是内部类,为了在内部类中保持对外部类的访问性,会增加一个指向外部类实例的字段。另外,在Java语言中字段无法重载,也就是字段名不能重复,即使两个字段的数据类型、修饰符都不相同。不过对于字节码来说,如果两个字段的描述符不一致,那么就可以有重复的字段名。

方法表

方法表结构与字段表结构几乎相同,仅在访问标志和属性表集合的可选项中有所区别。方法里的java代码存放在方法属性表中名为Code的属性中。
当描述符用来描述方法时,按照先参数列表,后返回值的顺序描述,参数列表按照参数的严格顺序放在一组小括号"()"内。比如方法void inc()的描述符是:()V。方法java.lang.String toString()的描述符是:()Ljava/lang/String。方法int indexOf(char[] source,int sourceOffset,int sourceCount,char[] target,int targetOffset,int targetCount,int fromIndex)的描述符是:([CII[CIII)I。
(扩展网上查的:
<init>
Java在编译之后会在字节码文件中生成<init>方法,称之为实例构造器,该实例构造器会将语句块,变量初始化,调用父类的构造器等操作收敛到<init>方法中,收敛顺序(这里只讨论非静态变量和语句块)为:

  1. 父类非静态变量初始化
  2. 父类非静态代码块
  3. 父类构造函数
  4. 子类非静态变量初始化
  5. 子类非静态代码块
  6. 子类构造函数
    所谓收敛到<init>方法中的意思就是,将这些操作放入到<init>中去执行

<clinit>
Java在编译之后会在字节码文件中生成<clinit>方法,称之为类构造器方法,在初始化阶段执行,会将静态语句块,静态变量初始化,收敛到<clinit>方法中,收敛顺序为:

  1. 父类静态变量初始化
  2. 父类静态代码块
  3. 子类静态变量初始化
  4. 子类静态代码块

<clinit>方法是在类加载过程中执行的,而<init>是在对象实例化执行的,所以<clinit>一定比<init>先执行。所以整个顺序就是:

  1. 父类静态变量初始化
  2. 父类静态代码块
  3. 子类静态变量初始化
  4. 子类静态代码块
  5. 父类变量初始化
  6. 父类代码块
  7. 父类构造函数
  8. 子类变量初始化
  9. 子类代码块
  10. 子类构造函数

属性表

属性表在前面出现了多次,在class文件、字段表和方法表都可以携带自己的属性表集合,来描述某些场景专有的信息。
属性表集合存在的位置也是不确定的,不仅可以存储在class文件结尾处,还可以作为数据项存在于类、方法表集合和字段表集合中。

常见的属性:

<table border="1" cellspacing="0" cellpadding="0"><tbody><tr><td valign="top">
<p><span style="font-size:14px;">属性名称</span></p>
</td>
<td valign="top">
<p><span style="font-size:14px;">使用位置</span></p>
</td>
<td valign="top">
<p><span style="font-size:14px;">含义</span></p>
</td>
<td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">Since</span></p>
</td>
</tr><tr><td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">Code</span></p>
</td>
<td valign="top">
<p><span style="font-size:14px;">方法表</span></p>
</td>
<td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">Java</span><span style="font-size:14px;">代码编译后的字节码指令</span></p>
</td>
<td valign="top">
<p><span style="font-size:14px;">《</span><span style="font-family:'Tw Cen MT';font-size:14px;">Java</span><span style="font-size:14px;">虚拟机规范(第二版)》</span></p>
</td>
</tr><tr><td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">ConstantValue</span></p>
</td>
<td valign="top">
<p><span style="font-size:14px;">字段表</span></p>
</td>
<td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">final</span><span style="font-size:14px;">关键字定义的常量值</span></p>
</td>
<td valign="top">
<p><span style="font-size:14px;">《</span><span style="font-family:'Tw Cen MT';font-size:14px;">Java</span><span style="font-size:14px;">虚拟机规范(第二版)》</span></p>
</td>
</tr><tr><td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">Deprecated</span></p>
</td>
<td valign="top">
<p><span style="font-size:14px;">类、方法表、字段表</span></p>
</td>
<td valign="top">
<p><span style="font-size:14px;">被声明为</span><span style="font-family:'Tw Cen MT';font-size:14px;">deprecated</span><span style="font-size:14px;">的方法和字段</span></p>
</td>
<td valign="top">
<p><span style="font-size:14px;">《</span><span style="font-family:'Tw Cen MT';font-size:14px;">Java</span><span style="font-size:14px;">虚拟机规范(第二版)》</span></p>
</td>
</tr><tr><td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">Exception</span></p>
</td>
<td valign="top">
<p><span style="font-size:14px;">方法表</span></p>
</td>
<td valign="top">
<p><span style="font-size:14px;">方法抛出的异常</span></p>
</td>
<td valign="top">
<p><span style="font-size:14px;">《</span><span style="font-family:'Tw Cen MT';font-size:14px;">Java</span><span style="font-size:14px;">虚拟机规范(第二版)》</span></p>
</td>
</tr><tr><td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">EnclosingMethod</span></p>
</td>
<td valign="top">
<p><span style="font-size:14px;">类文件</span></p>
</td>
<td valign="top">
<p><span style="font-size:14px;">仅当一个类为局部类或匿名类时才能拥有这个属性,来标识这个类所在的外围方法</span></p>
</td>
<td valign="top">
<p><span style="font-size:14px;">《</span><span style="font-family:'Tw Cen MT';font-size:14px;">Java</span><span style="font-size:14px;">虚拟机规范(第二版)》</span></p>
</td>
</tr><tr><td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">InnerClasses</span></p>
</td>
<td valign="top">
<p><span style="font-size:14px;">类文件</span></p>
</td>
<td valign="top">
<p><span style="font-size:14px;">内部类列表</span></p>
</td>
<td valign="top">
<p><span style="font-size:14px;">《</span><span style="font-family:'Tw Cen MT';font-size:14px;">Java</span><span style="font-size:14px;">虚拟机规范(第二版)》</span></p>
</td>
</tr><tr><td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">LineNumberTable</span></p>
</td>
<td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">Code</span><span style="font-size:14px;">属性</span></p>
</td>
<td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">Java</span><span style="font-size:14px;">源码的行号与字节码指令的对应关系</span></p>
</td>
<td valign="top">
<p><span style="font-size:14px;">《</span><span style="font-family:'Tw Cen MT';font-size:14px;">Java</span><span style="font-size:14px;">虚拟机规范(第二版)》</span></p>
</td>
</tr><tr><td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">LocalVariableTable</span></p>
</td>
<td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">Code</span><span style="font-size:14px;">属性</span></p>
</td>
<td valign="top">
<p><span style="font-size:14px;">方法的局部变量描述</span></p>
</td>
<td valign="top">
<p><span style="font-size:14px;">《</span><span style="font-family:'Tw Cen MT';font-size:14px;">Java</span><span style="font-size:14px;">虚拟机规范(第二版)》</span></p>
</td>
</tr><tr><td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">StackMapTable</span></p>
</td>
<td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">Code</span><span style="font-size:14px;">属性</span></p>
</td>
<td valign="top">
<p><span style="font-size:14px;">供新的类型检查验证器检查和处理目标方法的局部变量和操作数栈所需要的类型检查</span></p>
</td>
<td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">JDK 1.6</span></p>
</td>
</tr><tr><td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">Signature</span></p>
</td>
<td valign="top">
<p><span style="font-size:14px;">类、方法表、字段表</span></p>
</td>
<td valign="top">
<p><span style="font-size:14px;">用于保存泛型中的类型信息</span></p>
</td>
<td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">JDK 1.5</span></p>
</td>
</tr><tr><td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">SourceFile</span></p>
</td>
<td valign="top">
<p><span style="font-size:14px;">类文件</span></p>
</td>
<td valign="top">
<p><span style="font-size:14px;">记录源文件名称</span></p>
</td>
<td valign="top">
<p><span style="font-size:14px;">《</span><span style="font-family:'Tw Cen MT';font-size:14px;">Java</span><span style="font-size:14px;">虚拟机规范(第二版)》</span></p>
</td>
</tr><tr><td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">SourceDebugExtension</span></p>
</td>
<td valign="top">
<p><span style="font-size:14px;">类文件</span></p>
</td>
<td valign="top">
<p><span style="font-size:14px;">用于存储额外的调试信息</span></p>
</td>
<td valign="top">
<p><span style="font-size:14px;">《</span><span style="font-family:'Tw Cen MT';font-size:14px;">Java</span><span style="font-size:14px;">虚拟机规范(第二版)》</span></p>
</td>
</tr><tr><td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">Synthetic</span></p>
</td>
<td valign="top">
<p><span style="font-size:14px;">类、方法表、字段表</span></p>
</td>
<td valign="top">
<p><span style="font-size:14px;">标识方法或字段为编译器自动生成</span></p>
</td>
<td valign="top">
<p><span style="font-size:14px;">《</span><span style="font-family:'Tw Cen MT';font-size:14px;">Java</span><span style="font-size:14px;">虚拟机规范(第二版)》</span></p>
</td>
</tr><tr><td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">LocalVariableTypeTable</span></p>
</td>
<td valign="top">
<p><span style="font-size:14px;">类</span></p>
</td>
<td valign="top">
<p><span style="font-size:14px;">使用特征签名代替描述符,为了描述泛型参数化类型</span></p>
</td>
<td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">JDK 1.5</span></p>
</td>
</tr><tr><td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">RuntimeVisibleAnnotations</span></p>
</td>
<td valign="top">
<p><span style="font-size:14px;">类、方法表、字段表</span></p>
</td>
<td valign="top">
<p><span style="font-size:14px;">为动态注解提供支持</span></p>
</td>
<td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">JDK 1.5</span></p>
</td>
</tr><tr><td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">RuntimeInvisibleAnnotations</span></p>
</td>
<td valign="top">
<p><span style="font-size:14px;">类、方法表、字段表</span></p>
</td>
<td valign="top">
<p><span style="font-size:14px;">与</span><span style="font-family:'Tw Cen MT';font-size:14px;">RuntimeVisibleAnnotations</span><span style="font-size:14px;">作用相反</span></p>
</td>
<td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">JDK 1.5</span></p>
</td>
</tr><tr><td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">RuntimeVisbleParameterAnnotations</span></p>
</td>
<td valign="top">
<p><span style="font-size:14px;">方法表</span></p>
</td>
<td valign="top">
<p><span style="font-size:14px;">与</span><span style="font-family:'Tw Cen MT';font-size:14px;">RuntimeVisibleAnnotations</span><span style="font-size:14px;">类似</span></p>
</td>
<td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">JDK 1.5</span></p>
</td>
</tr><tr><td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">RuntimeInvisbleParameterAnnotations</span></p>
</td>
<td valign="top">
<p><span style="font-size:14px;">方法表</span></p>
</td>
<td valign="top">
<p><span style="font-size:14px;">与</span><span style="font-family:'Tw Cen MT';font-size:14px;">RuntimeInvisibleAnnotations</span><span style="font-size:14px;">类似</span></p>
</td>
<td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">JDK 1.5</span></p>
</td>
</tr><tr><td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">AnnotationDefault</span></p>
</td>
<td valign="top">
<p><span style="font-size:14px;">方法表</span></p>
</td>
<td valign="top">
<p><span style="font-size:14px;">用于记录注解类元素的默认值</span></p>
</td>
<td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">JDK 1.5</span></p>
</td>
</tr><tr><td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">BootstrapMethods</span></p>
</td>
<td valign="top">
<p><span style="font-size:14px;">类文件</span></p>
</td>
<td valign="top">
<p><span style="font-size:14px;">用于保存</span><span style="font-family:'Tw Cen MT';font-size:14px;">invokedynamic</span><span style="font-size:14px;">指令引用的引导方法限定符</span></p>
</td>
<td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">JDK 1.7</span></p>
</td>
</tr></tbody></table>

<p>由于属性表集合的限制较小,每个属性都会有自己的格式,因此class文件对于属性的格式要求也比较宽松,只需要满足一些特定的条件即可。下表是属性的结构:</p>
<p>
</p><table border="1" cellspacing="0" cellpadding="0"><tbody><tr><td valign="top">
<p><span style="font-size:14px;">类型</span></p>
</td>
<td valign="top">
<p><span style="font-size:14px;">名称</span></p>
</td>
<td valign="top">
<p><span style="font-size:14px;">数量</span></p>
</td>
</tr><tr><td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">U2</span></p>
</td>
<td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">attribute_name_index</span></p>
</td>
<td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">1</span></p>
</td>
</tr><tr><td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">U4</span></p>
</td>
<td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">attribute_length</span></p>
</td>
<td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">1</span></p>
</td>
</tr><tr><td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">U1</span></p>
</td>
<td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">info</span></p>
</td>
<td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">attribute_length</span></p>
</td>
</tr></tbody></table>

<p>(1)Code属性</p>
<p>最常用的属性恐怕就是Code属性了,因为大多数的方法都会有编译后的字节码指令,这些指令就存储在方法表中的Code属性中。如果一个Java程序的信息可以分为代码(方法体中的代码)和元数据(包括类、字段、方法定义以及其它信息),那么Code属性存储的就是代码,其它所有的结构存储的都是元数据。不过并非所有的方法表都有这个Code属性,比如接口或抽象类中的方法表就不存在Code属性(JDK 1.8中的接口也可以定义方法了)。Code属性的结构如下:</p>
<p>
</p><table border="1" cellspacing="0" cellpadding="0"><tbody><tr><td valign="top">
<p><span style="font-size:14px;">类型</span></p>
</td>
<td valign="top">
<p><span style="font-size:14px;">名称</span></p>
</td>
<td valign="top">
<p><span style="font-size:14px;">数量</span></p>
</td>
</tr><tr><td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">U2</span></p>
</td>
<td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">attribute_name_index</span></p>
</td>
<td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">1</span></p>
</td>
</tr><tr><td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">U4</span></p>
</td>
<td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">attribute_lenght</span></p>
</td>
<td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">1</span></p>
</td>
</tr><tr><td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">U2</span></p>
</td>
<td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">max_stack</span></p>
</td>
<td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">1</span></p>
</td>
</tr><tr><td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">U2</span></p>
</td>
<td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">max_locals</span></p>
</td>
<td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">1</span></p>
</td>
</tr><tr><td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">U4</span></p>
</td>
<td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">code_length</span></p>
</td>
<td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">1</span></p>
</td>
</tr><tr><td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">U1</span></p>
</td>
<td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">code</span></p>
</td>
<td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">code_length</span></p>
</td>
</tr><tr><td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">U2</span></p>
</td>
<td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">exception_table_length</span></p>
</td>
<td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">1</span></p>
</td>
</tr><tr><td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">exception_info</span></p>
</td>
<td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">exception_table</span></p>
</td>
<td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">exception_table_length</span></p>
</td>
</tr><tr><td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">U2</span></p>
</td>
<td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">attributes_count</span></p>
</td>
<td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">1</span></p>
</td>
</tr><tr><td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">attribute_info</span></p>
</td>
<td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">attributes</span></p>
</td>
<td valign="top">
<p><span style="font-family:'Tw Cen MT';font-size:14px;">attributes_count</span></p>
</td>
</tr></tbody></table>
(2)LinNumberTable

(3)LocalVariableTable
(4) SourceFile

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,686评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,668评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,160评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,736评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,847评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,043评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,129评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,872评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,318评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,645评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,777评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,470评论 4 333
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,126评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,861评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,095评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,589评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,687评论 2 351

推荐阅读更多精彩内容