《Java核心技術》 Class類文件結構

運行機制

開篇

源代碼HelloWorld.java

/**
 * Created by jack on 2017/3/16.
 *
 * @author jack
 * @date 2017/03/16
 */
public class HelloWorld {
    public static void main(String[] args){
        System.out.println("Hello,World");
    }
}

編譯成HelloWorld.class

cafe babe 0000 0031 0022 0a00 0600 1409
0015 0016 0800 170a 0018 0019 0700 1a07
001b 0100 063c 696e 6974 3e01 0003 2829
5601 0004 436f 6465 0100 0f4c 696e 654e
756d 6265 7254 6162 6c65 0100 124c 6f63
616c 5661 7269 6162 6c65 5461 626c 6501
0004 7468 6973 0100 0c4c 4865 6c6c 6f57
6f72 6c64 3b01 0004 6d61 696e 0100 1628
5b4c 6a61 7661 2f6c 616e 672f 5374 7269
6e67 3b29 5601 0004 6172 6773 0100 135b
4c6a 6176 612f 6c61 6e67 2f53 7472 696e
673b 0100 0a53 6f75 7263 6546 696c 6501
000f 4865 6c6c 6f57 6f72 6c64 2e6a 6176
610c 0007 0008 0700 1c0c 001d 001e 0100
0b48 656c 6c6f 2c57 6f72 6c64 0700 1f0c
0020 0021 0100 0a48 656c 6c6f 576f 726c
6401 0010 6a61 7661 2f6c 616e 672f 4f62
6a65 6374 0100 106a 6176 612f 6c61 6e67
2f53 7973 7465 6d01 0003 6f75 7401 0015
4c6a 6176 612f 696f 2f50 7269 6e74 5374
7265 616d 3b01 0013 6a61 7661 2f69 6f2f
5072 696e 7453 7472 6561 6d01 0007 7072
696e 746c 6e01 0015 284c 6a61 7661 2f6c
616e 672f 5374 7269 6e67 3b29 5600 2100
0500 0600 0000 0000 0200 0100 0700 0800
0100 0900 0000 2f00 0100 0100 0000 052a
b700 01b1 0000 0002 000a 0000 0006 0001
0000 0007 000b 0000 000c 0001 0000 0005
000c 000d 0000 0009 000e 000f 0001 0009
0000 0037 0002 0001 0000 0009 b200 0212
03b6 0004 b100 0000 0200 0a00 0000 0a00
0200 0000 0900 0800 0a00 0b00 0000 0c00
0100 0000 0900 1000 1100 0000 0100 1200
0000 0200 13

這就是jvm的byte code字節(jié)碼。這一串“天書”,如果不看JVM虛擬機規(guī)范的人,正常的智商,是看不懂的。

想看懂這一串十六進制的字符,就要了解class文件協(xié)議。

第一行釋義:

magic:魔數,0xCAFEBABE(cafe babe)
minor_version:占2字節(jié),次版本號,0x0000
majro_version:占2字節(jié),主版本號,0x0031,轉化為十進制為49,是使用JDK1.5編譯的(JDK1.5:0x0031,JDK1.6:0x0032,JDK1.7:0x0033)
高版本的JDK可以向下兼容以前版本的Class文件,但是無法運行以后版本的Class文件,即使文件格式并未發(fā)生變化
如果使用JDK1.5運行使用JDK1.6編譯的Class文件,會報:

java.lang.UnsupportedClassVersionError: Bad version number in .class file  

就是由于JDK1.6編譯的文件版本號超過了JDK1.5虛擬機所接受的范圍

Java class文件是二進制文件。為了便于理解它,JVM提供者提供了javap,反匯編器(編譯,反編譯的本質,是“映射”)。

一切皆是映射(光劍)

使用javap產生的結果是Java匯編語言。在上面的例子中,下面的Java匯編代碼是通過:

javap -c HelloWorld.class

進行反匯編得到的。

$ javap -c HelloWorld.class 
Compiled from "HelloWorld.java"
public class HelloWorld {
  public HelloWorld();
    Code:
       0: aload_0
       1: invokespecial #1                  // Method java/lang/Object."<init>":()V
       4: return

  public static void main(java.lang.String[]);
    Code:
       0: getstatic     #2                  // Field java/lang/System.out:Ljava/io/PrintStream;
       3: ldc           #3                  // String Hello,World
       5: invokevirtual #4                  // Method java/io/PrintStream.println:(Ljava/lang/String;)V
       8: return
}


所以說,如果你足夠牛逼,不需要使用java語言寫源碼,直接使用jvm指令集寫jvm的匯編語言。這樣,什么Scala,Java,Kotlin,Clojure這些運行在jvm上的高級編程語言,對你來說,真的就是浮云了。你也不需要糾結什么OOP,FP,Design Pattern, etc.

Class類文件結構

class文件的結構

簡單講,jvm的class文件是針對jvm的私家定制。只要運行在jvm上byte code,都需要按照class文件format來。否則jvm解析不了。自然無法執(zhí)行。

官網文檔: The Java Virtual Machine class file format

class文件時java虛擬機執(zhí)行引擎的數據入口,也是java技術體系的基礎支柱之一,了解class文件的結構對后面進一步了解虛擬機執(zhí)行引擎有很重要的意義。

class文件是一組以八位字節(jié)為基礎單位的二進制流,各個數據項目嚴格按照順序緊湊地排列在class文件中,中間沒有添加任何分隔符,這使得整個class文件中存儲的內容幾乎全部都是程序運行的必要數據,沒有空隙存在。當需要占用8位字節(jié)以上的空間數據時,則會按照高位在前的方式分割成若干個8位字節(jié)進行存儲。

ClassFile數據結構

ClassFile {  
    u4 magic;//魔數(0xCAFEBABE)  
    u2 minor_version;//次版本號  
    u2 major_version;//主版本號  
    u2 constant_pool_count;//常量池容量計數值  
    cp_info constant_pool[constant_pool_count-1];//常量池  
    u2 access_flags;//訪問標志  
    u2 this_class;//類索引  
    u2 super_class;//父類索引  
    u2 interfaces_count;//接口計數器  
    u2 interfaces[interfaces_count];//接口索引集合  
    u2 fields_count;//字段計數器  
    field_info fields[fields_count];//字段表  
    u2 methods_count;//方法計數器  
    method_info methods[methods_count];//方法表  
    u2 attributes_count;//屬性表計數器  
    attribute_info attributes[attributes_count];//屬性表集合  
}  

Class文件中,類的全限定名、字段、方法都是使用CONSTANT_Utf8_info類型常量來描述名稱,而該常量的長度由2個字節(jié)表示,所以類的全限定名、字段名、方法名的最大長度不能超過2個字節(jié)所能表示的最大整數,也就是65535.

博文參考:

http://blog.csdn.net/a19881029/article/details/16117251

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯(lián)系作者
【社區(qū)內容提示】社區(qū)部分內容疑似由AI輔助生成,瀏覽時請結合常識與多方信息審慎甄別。
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發(fā)布,文章內容僅代表作者本人觀點,簡書系信息發(fā)布平臺,僅提供信息存儲服務。

相關閱讀更多精彩內容

友情鏈接更多精彩內容