看了第6章的java class文件这一部分,我觉得对class文件有进一步的了解。于是想作一点笔记,以备以后查时方便。

这章的一个例子虽然简单,但是我觉的很有一定代表性。例子如下:

1:class Act {
2:   public static void doMathForever(int m) {
3:      int i = 0;
4:     int n=4>3?1:0;
5:        for (;;) {
6:           i += 1;
7:            i *= 2;
8:       }
9:    }
10:}

我用javap -verbose Act查看,代码如下:(红色为自己加上的注解)

Compiled from "Act.java"

因为编译原文件不一定必须与java语言相关,可以使用其他语言来编写程序,然后将其编译为class文件,所以这里要写上Act.java而不是Act. 呵呵,想来以前为什么javac 来编译源文件时一定要加.java,而java 执行时就不需要加.java了,当时就是搞不懂,现在明白了。

class Act extends java.lang.Object
SourceFile: "Act.java"

SourceFile属性,它提供了产生class文件的源文件的名称,它是一个可选的项,为什么说它是可选项的,因为class可以自己用UE等工具写。

minor version: 0

major version: 49

版本号:我用的是5.0的,出来的major version 是49,书上说1.0或1.2是45,1.4我想可能是48 吧。呵呵,还没有试过。

Constant pool:

这个是常量池,刚看书的时候我还以为常量池放的一定是常量。错。

常量池就是该类型所用常量的一个有序集合,包括直接常量(String,integer,floating  point常量)和对其他类型、字段和方法的符号引用。const #1 = Method #3.#12; //  java/lang/Object."":()V

因为Constant pool#0是不用的,所以只能从#1开始,但不知为什么第一个是Method而不是class,我现在还不明白。

const #2 = class #13; //  Act
CONSTANT_Class是对一个类或接口的符号引用。这个是指向#13
const #3 = class #14; //  java/lang/Object
const #4 = Asciz ;

这个Asciz应该表示是CONSTANT_Utf8吧,因为C ONSTANT_Utf8可以是存储四种基本信息类型:文字字符串、被定义的类和接口描述、对其他类或接口的符号引用以及属性相关的字符串。

是实例的初始化方法,是实例调用方法前必须要的,

是类的初始化方法。const #5 = Asciz ()V;

此的描述符,()V表示没有带参数,返回的是void型的。

书上介绍:

()I  int getSize()
() Ljava/lang/String      String toString()
const #6 = Asciz Code;

Code在所有不是抽象或者本地方法的method_info信息中,都存在一个Code_attribute表。

但是我试一下在抽象的方法中同样也有code的啊,不懂中。const #7 = Asciz LineNumberTable;

LineNumberTable属性建立了方法字节码流偏移量和源代码行号之间的映射关系。关于这一点,我在下面详细谈到。const #8 = Asciz doMathForever;

#8 字段名和方法名以简单名称出现在常量池的入口中,存的时候存简单名字,调用的时候要加 上全限定名const #9 = Asciz (I)V;

doMathForover的描述符,参数是int类型,返回值是void类型const #10 = Asciz SourceFile;

const #11 = Asciz Act.java;

const #12 = NameAndType #4:#5;//  "":()V

NameAndType指向字段或者方法的符号引用的一部分。为什么没有指向#8,#9呢?不懂中。const #13 = Asciz Act;

const #14 = Asciz java/lang/Object;

#13,#14 CONSTANT_Utf8_info 表的入口,容纳了 类/接口全限定名等信息。用“/”代替“.”。在class文件的this_class字段是对常量池的一个引用,指向了常量池中CONSTANT_ClASS_info表,该表由两项组成,即标签和name_index。在这个例子里,this_class的值为2,也就是常量池里的第二项,这项标签是class,name_index是13,即指向常量池的第13项。第13项里就有当前类的全限定名。

{
Act();
Code:
Stack=1, Locals=1, Args_size=1

需要的栈为1个,只是this的存取;本地变量为1个,也是this,方法的参数是1个,也是this传入。   0: aload_0

从局部变量0中装载引用类型。

1: invokespecial #1; //Method java/lang/Object."":()V

调用Object的方法,因为每个类都是继承Object,所以实例时都要调用Object的方法。

4: return
LineNumberTable:
line 1: 0
public static void doMathForever(int);
Code:
Stack=2, Locals=3, Args_size=1
0: iconst_0

将i 压入栈   1: istore_1

放i   2: iconst_1

同上   3: istore_2

4: iinc 1, 1  i++

7   iload_1 从局部变量1中装载int类型值。

8: iconst_2

9: imul

10: istore_1

11: goto 4

LineNumberTable:

line 3: 0  源代码第3行对应Code中的0偏移量。

line 4: 2

line 6: 4

line 7: 7

line 8: 11

}