分享
 
 
 

Class文件详解 (1)

王朝java/jsp·作者佚名  2006-01-09
窄屏简体版  字體: |||超大  

作者:asklxf 文章来源:http://www.j2medev.com/Article/ShowArticle.asp?ArticleID=105

我们都知道,Java编译器负责将.java文件编译成.class文件,class文件存储的是java字节码,与.java文件无关(只要你愿意写一个编译器,也可以将别的语言写的源代码编译成.class文件),本文准备详细解剖class文件的内部结构,并且把class文件结构读取并显示出来。

Class文件的格式由JVM规范规定,一共有以下部分:

1. magic number,必须是0xCAFEBABE,用于快速识别是否是一个class文件。

2. version,包括major和minor,如果版本号超过了JVM的识别范围,JVM将拒绝执行。

3. constant pool,常量池,存放所有用到的常量。

4. access flag,定义类的访问权限。

5. this class和super class,指示如何找到this class和super class。

6. interfaces,存放所有interfaces。

7. fields,存放所有fields。

8. methods,存放所有methods。

9. attributes,存放所有attributes。

先写一个Test.java:

package example.test;

public final class TestClass {

public int id = 1234567;

public void test() {}

}

然后编译,放在C:\example\test\Test.class。

我们用Java来读取和分析class,ClassAnalyzer的功能便是读取Test.class,分析结构,然后显示出来:

package classfile.format;

import java.io.*;

public class ClassAnalyzer {

public static void main(String[] args) {

DataInputStream input = null;

try {

input = new DataInputStream(new BufferedInputStream(new FileInputStream(

"C:\\example\\test\\TestClass.class"

)));

analyze(input);

}

catch(Exception e) {

System.out.println("Analyze failed!");

}

finally {

try { input.close(); } catch(Exception e) {}

}

}

public static void analyze(DataInputStream input) throws IOException {

// read magic number:

int magic = input.readInt();

if(magic==0xCAFEBABE)

System.out.println("magic number = 0xCAFEBABE");

else

throw new RuntimeException("Invalid magic number!");

// read minor version and major version:

short minor_ver = input.readShort();

short major_ver = input.readShort();

System.out.println("Version = " + major_ver + "." + minor_ver);

// read constant pool:

short const_pool_count = input.readShort();

System.out.println("constant pool size = " + const_pool_count);

// read each constant:

for(int i=1; i<const_pool_count; i++) {

analyzeConstant(input, i);

}

}

public static void analyzeConstant(DataInputStream input, int index) throws IOException {

byte flag = input.readByte();

// for read:

byte n8;

short n16;

int n32;

long n64;

float f;

double d;

byte[] buffer;

System.out.println("\nconst index = " + index + ", flag = " + (int)flag);

switch(flag) {

case 1: // utf-8 string

System.out.println(" const type = Utf8");

n16 = input.readShort();

System.out.println(" length = " + n16);

buffer = new byte[n16];

input.readFully(buffer);

System.out.println(" value = " + new String(buffer));

break;

case 3: // integer

System.out.println(" const type = Integer");

n32 = input.readInt();

System.out.println(" value = " + n32);

break;

case 4: // float

System.out.println(" const type = Float");

f = input.readFloat();

System.out.println(" value = " + f);

break;

case 5: // long

System.out.println(" const type = Long");

n64 = input.readLong();

System.out.println(" value = " + n64);

break;

case 6: // double

System.out.println(" const type = Double");

d = input.readDouble();

System.out.println(" value = " + d);

break;

case 7: // class or interface reference

System.out.println(" const type = Class");

n16 = input.readShort();

System.out.println(" index = " + n16 + " (where to find the class name)");

break;

case 8: // string

System.out.println(" const type = String");

n16 = input.readShort();

System.out.println(" index = " + n16);

break;

case 9: // field reference

System.out.println(" const type = Fieldref");

n16 = input.readShort();

System.out.println("class index = " + n16 + " (where to find the class)");

n16 = input.readShort();

System.out.println("nameAndType = " + n16 + " (where to find the NameAndType)");

break;

case 10: // method reference

System.out.println(" const type = Methodref");

n16 = input.readShort();

System.out.println("class index = " + n16 + " (where to find the class)");

n16 = input.readShort();

System.out.println("nameAndType = " + n16 + " (where to find the NameAndType)");

break;

case 11: // interface method reference

System.out.println(" const type = InterfaceMethodref");

n16 = input.readShort();

System.out.println("class index = " + n16 + " (where to find the interface)");

n16 = input.readShort();

System.out.println("nameAndType = " + n16 + " (where to find the NameAndType)");

break;

case 12: // name and type reference

System.out.println(" const type = NameAndType");

n16 = input.readShort();

System.out.println(" name index = " + n16 + " (where to find the name)");

n16 = input.readShort();

System.out.println(" descripter = " + n16 + " (where to find the descriptor)");

break;

default:

throw new RuntimeException("Invalid constant pool flag: " + flag);

}

}

}

输出结果为:

magic number = 0xCAFEBABE

Version = 48.0

constant pool size = 22

const index = 1, flag = 1

const type = Utf8

length = 22

value = example/test/TestClass

const index = 2, flag = 7

const type = Class

index = 1 (where to find the class name)

const index = 3, flag = 1

const type = Utf8

length = 16

value = java/lang/Object

const index = 4, flag = 7

const type = Class

index = 3 (where to find the class name)

const index = 5, flag = 1

const type = Utf8

length = 2

value = id

const index = 6, flag = 1

const type = Utf8

length = 1

value = I

const index = 7, flag = 1

const type = Utf8

length = 6

value = <init>

const index = 8, flag = 1

const type = Utf8

length = 3

value = ()V

const index = 9, flag = 1

const type = Utf8

length = 4

value = Code

const index = 10, flag = 12

const type = NameAndType

name index = 7 (where to find the name)

descripter = 8 (where to find the descriptor)

const index = 11, flag = 10

const type = Methodref

class index = 4 (where to find the class)

nameAndType = 10 (where to find the NameAndType)

const index = 12, flag = 3

const type = Integer

value = 1234567

const index = 13, flag = 12

const type = NameAndType

name index = 5 (where to find the name)

descripter = 6 (where to find the descriptor)

const index = 14, flag = 9

const type = Fieldref

class index = 2 (where to find the class)

nameAndType = 13 (where to find the NameAndType)

const index = 15, flag = 1

const type = Utf8

length = 15

value = LineNumberTable

const index = 16, flag = 1

const type = Utf8

length = 18

value = LocalVariableTable

const index = 17, flag = 1

const type = Utf8

length = 4

value = this

const index = 18, flag = 1

const type = Utf8

length = 24

value = Lexample/test/TestClass;

const index = 19, flag = 1

const type = Utf8

length = 4

value = test

const index = 20, flag = 1

const type = Utf8

length = 10

value = SourceFile

const index = 21, flag = 1

const type = Utf8

length = 14

value = TestClass.java

我们暂时只读取到constant pool,后面的信息下次再继续 :)

参考:《Inside the Java Virture Mechine》

 
 
 
免责声明:本文为网络用户发布,其观点仅代表作者个人观点,与本站无关,本站仅提供信息存储服务。文中陈述内容未经本站证实,其真实性、完整性、及时性本站不作任何保证或承诺,请读者仅作参考,并请自行核实相关内容。
2023年上半年GDP全球前十五强
 百态   2023-10-24
美众议院议长启动对拜登的弹劾调查
 百态   2023-09-13
上海、济南、武汉等多地出现不明坠落物
 探索   2023-09-06
印度或要将国名改为“巴拉特”
 百态   2023-09-06
男子为女友送行,买票不登机被捕
 百态   2023-08-20
手机地震预警功能怎么开?
 干货   2023-08-06
女子4年卖2套房花700多万做美容:不但没变美脸,面部还出现变形
 百态   2023-08-04
住户一楼被水淹 还冲来8头猪
 百态   2023-07-31
女子体内爬出大量瓜子状活虫
 百态   2023-07-25
地球连续35年收到神秘规律性信号,网友:不要回答!
 探索   2023-07-21
全球镓价格本周大涨27%
 探索   2023-07-09
钱都流向了那些不缺钱的人,苦都留给了能吃苦的人
 探索   2023-07-02
倩女手游刀客魅者强控制(强混乱强眩晕强睡眠)和对应控制抗性的关系
 百态   2020-08-20
美国5月9日最新疫情:美国确诊人数突破131万
 百态   2020-05-09
荷兰政府宣布将集体辞职
 干货   2020-04-30
倩女幽魂手游师徒任务情义春秋猜成语答案逍遥观:鹏程万里
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案神机营:射石饮羽
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案昆仑山:拔刀相助
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案天工阁:鬼斧神工
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案丝路古道:单枪匹马
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案镇郊荒野:与虎谋皮
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案镇郊荒野:李代桃僵
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案镇郊荒野:指鹿为马
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案金陵:小鸟依人
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案金陵:千金买邻
 干货   2019-11-12
 
推荐阅读
 
 
 
>>返回首頁<<
 
靜靜地坐在廢墟上,四周的荒凉一望無際,忽然覺得,淒涼也很美
© 2005- 王朝網路 版權所有