解析Java的Class文件格式——解析魔数和版本号(一)

时间:2022-10-30 17:19:06

解析JavaClass文件格式——解析魔数和版本号()

作者:陈跃峰

出自:http://blog.csdn.net/mailbomb

 

       熟悉Java语言有好几年了,技术也学了一些,现在主要从事J2ME技术方面的工作,最近工作不是很忙,就找了本电子书——《深入Java虚拟机(第二版)》,仔细阅读了一下,读起来比较吃力,现在把Java class文件格式的读书笔记共享给大家。

       众所周知,Java语言的可执行文件是class文件,俗称类文件。这个文件为了让不同平台的虚拟机都能够正确的解释,详细规定了其文件格式。下面就按照顺序进行介绍:

1、  魔数(magic)

为了方便虚拟机识别文件是否是class格式的文件,SUN公司规定每个class文件都必须以一个word(4个字节)来开始,这个数字就称为魔数。魔数是有4个字节的无符号数字构成的,而且规定魔数必须是0xCAFEBABE

2、  版本号(version)

魔数后续的内容就是一个word的长度来表示生成的class文件的版本号,版本号分为主版本号和次版本号,其中前两个字节表示次版本号,后两个字节表示主版本号,排列的顺序遵从高位在前,低位在后的原则。

 

       下面我写了一个很简单的示例代码,从一个名字为First.class文件中读取到这些信息,然后输出出来,不足之处请大家指正!

       该程序的代码如下:

import java.io.*;

 

/**

 * 解析class文件格式

 */

public class ParseClassFile{

       public static void main(String args[]){

             

              try{

                     //读取文件数据,文件是当前目录下的First.class

                     FileInputStream fis = new FileInputStream("./First.class");

                    

                     int length = fis.available();

                     //文件数据

                     byte[] data = new byte[length];

                    

                     //读取文件到字节数组

                     fis.read(data);

                    

                     //关闭文件

                     fis.close();

                    

                     //解析文件数据

                     parseFile(data);

                    

              }catch(Exception e){

                     System.out.println(e);   

              }    

       }

      

      

       private static void parseFile(byte[] data){

              //输出魔数

              System.out.print("魔数(magic):0x");

              System.out.print(Integer.toHexString(data[0]).substring(6).toUpperCase());

              System.out.print(Integer.toHexString(data[1]).substring(6).toUpperCase());

              System.out.print(Integer.toHexString(data[2]).substring(6).toUpperCase());

              System.out.println(Integer.toHexString(data[3]).substring(6).toUpperCase());

             

              //主版本号和次版本号码

              int minor_version = (((int)data[4]) << 8) + data[5];

              int major_version = (((int)data[6]) << 8) + data[7];

             

              System.out.println("版本号(version):" + major_version + "." + minor_version);

       }    

}