深入解析Class類文件的結(jié)構(gòu)

前言

要深入學(xué)習(xí)Java以及Java虛擬機(jī)搔弄,深入學(xué)習(xí)Java字節(jié)碼文件是繞不開的一條路日熬,只有知道了字節(jié)碼文件里的排列結(jié)構(gòu)携冤,你才能透徹的了解在JVM里,類加載是怎么加載Java類的须床,是怎么將二進(jìn)制流轉(zhuǎn)化為運(yùn)行時(shí)數(shù)據(jù)結(jié)構(gòu)的铐料。

Class文件是是一組以8字節(jié)為基礎(chǔ)單位的二進(jìn)制流,各個(gè)數(shù)據(jù)項(xiàng)目嚴(yán)格按照順序緊湊地排列在Class文件中豺旬,中間沒有任何分隔符。

這里的Class文件其實(shí)不是特指Java的字節(jié)碼文件柒凉,任何編程語(yǔ)言的編譯器只要按照字節(jié)碼文件規(guī)范編譯成Class文件族阅,都可以在JVM上運(yùn)行,所以字節(jié)碼文件和JVM是和語(yǔ)言無(wú)關(guān)的膝捞。

另外一般Class文件指的不一定是存儲(chǔ)在磁盤上的以.class后綴結(jié)束的文件坦刀,是一種泛指,指的是一切按照字節(jié)碼文件規(guī)范排列的二進(jìn)制字節(jié)流。

字節(jié)碼文件解析

Class文件采用下面這種類似C語(yǔ)言的結(jié)構(gòu)體的偽結(jié)構(gòu)來(lái)存儲(chǔ)數(shù)據(jù)鲤遥,整個(gè)Class文件是一張表沐寺,表里又由無(wú)符號(hào)數(shù)和表組成。

ClassFile { 
    u4  magic; // 魔數(shù)盖奈,固定為"0xCAFEBABY"
    u2  minor_version; //jdk次版本號(hào)
    u2  major_version;  //jdk主版本號(hào)
    u2  constant_pool_count;  //常量池?cái)?shù)組大小混坞,從1計(jì)數(shù)
    cp_info  constant_pool[constant_pool_count - 1]; //常量池?cái)?shù)組
    u2  access_flags;  //類的訪問(wèn)標(biāo)志,如:public
    u2  this_class;  //類索引钢坦,指向常量池中的類符號(hào)引用
    u2  super_class;  //父類索引究孕,指向常量池中的類符號(hào)引用
    u2  interfaces_count; //實(shí)現(xiàn)的接口的數(shù)量
    u2  interfaces[interfaces_count]; //接口列表,按implements后面的接口順序
    u2  fields_count;  //字段數(shù)
    field_info  fields[fields_count]; //字段表
    u2  methods_count; //方法數(shù)
    method_info  methods[methods_count]; //方法表
    u2  attributes_count; //屬性表大小
    attribute_info  attributes[attributes_count]; //屬性表
}

從上面的偽結(jié)構(gòu)可以看到爹凹,Class文件根據(jù)上面的順序把規(guī)定的數(shù)據(jù)類型按照占用的字節(jié)依次排列下來(lái)厨诸。

下面通過(guò)一個(gè)例子來(lái)實(shí)戰(zhàn)分析一下Class文件

//Test.class
public class Test { 
    public static int a = 1;
    public static final int b = 1; 
    public void say(){
        System.out.println("Hello");
    }
}

字節(jié)碼文件實(shí)戰(zhàn)分析.png

上圖是編譯后的Test.class文件的二進(jìn)制數(shù)據(jù),可以按照上面ClassFile的結(jié)構(gòu)順序依次分析下禾酱,下面是部分分析結(jié)果:
(1) u4 magic
????4個(gè)字節(jié)(000h:0123)魔數(shù): 0xCAFEBABY

(2) u2 minor_version
????2個(gè)字節(jié)(000h:45)次版本號(hào): 0x0000, 次版本號(hào)為0

(3) u2 major_version
????2個(gè)字節(jié)(000h:67)主版本號(hào): 0x0034,即52,JDK1.0-1.1:45.0 ~ 45.3, 1.1后版本增1微酬,數(shù)字加1,所以這里用的是1.1 + 0.(52-45) = 1.8

(4) u2 constant_pool_count
????2個(gè)字節(jié)(000h:89)常量池大小:0x0027,即39颤陶,常量池?cái)?shù)組是從1開始計(jì)數(shù)的颗管,說(shuō)明常量池中有38個(gè)常量,后面依次排列的就是常量池的38個(gè)常量

(5) cp_info constant_pool[constant_pool_count - 1]
????常量池所占的字節(jié)數(shù)是由常量池中常量的數(shù)量以及類型所決定的指郁,這里有38個(gè)常量忙上,每個(gè)常量開頭都有一個(gè)字節(jié)的tag標(biāo)識(shí)常量的類型,具體類型可以參考最下面的腦圖闲坎,根據(jù)這個(gè)標(biāo)識(shí)可以找到這個(gè)常量所占的字節(jié)以及含義疫粥,下面分析其中一個(gè)常量,其余的讀者有興趣可以全部完成

  • 000h:a 0x0A,表示常量類型為10腰懂,查表可知是CONSTANT_Methodref方法符號(hào)引用梗逮,那接下來(lái)的四個(gè)字節(jié),前兩個(gè)字節(jié)表示指向常量池中方法所在類的符號(hào)引用的索引項(xiàng)绣溜,就是常量池的數(shù)組下標(biāo)慷彤,所在的位置是方法所在類的符號(hào)引用
  • 000h:bc 0x0007,指向常量池?cái)?shù)組第7個(gè)元素,第7個(gè)常量是一個(gè)java.lang.Object類的符號(hào)引用
  • 000h:de 0x0018, 指向常量池?cái)?shù)組的第24個(gè)元素怖喻,第24個(gè)常量是一個(gè)名稱和類型的符號(hào)引用底哗,方法名是<init>,描述符是()V
    這樣第一個(gè)常量就分析完成锚沸,共占5個(gè)字節(jié)跋选,表示的是方法符號(hào)引用,該方法所在的類是Object類哗蜈,方法名稱是<init>, 無(wú)參數(shù)前标,返回值是void

借助工具javap可以更直觀的看到我們剛剛分析的部分結(jié)果以及全部類文件的結(jié)構(gòu)坠韩,使用以下命令即可:

javap -v Test.class

結(jié)果如圖:

javap.png

通過(guò)上面的圖可以看到,和我們上面的部分分析是一致的

Class文件結(jié)構(gòu)腦圖

下面是我在看《深入理解Java虛擬機(jī)》這本書的時(shí)候整理的關(guān)于Class文件結(jié)構(gòu)的腦圖炼列,圖片比較大只搁,右鍵另存為圖片再查看會(huì)更方便。

class.png

更多干貨俭尖,請(qǐng)掃描關(guān)注公眾號(hào)氢惋,持續(xù)更新


image.png

原文鏈接:http://www.jackielee.cn/posts/7eb7d5c7.html

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市目溉,隨后出現(xiàn)的幾起案子明肮,更是在濱河造成了極大的恐慌,老刑警劉巖缭付,帶你破解...
    沈念sama閱讀 221,430評(píng)論 6 515
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件柿估,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡陷猫,警方通過(guò)查閱死者的電腦和手機(jī)秫舌,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 94,406評(píng)論 3 398
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)绣檬,“玉大人足陨,你說(shuō)我怎么就攤上這事〗课矗” “怎么了墨缘?”我有些...
    開封第一講書人閱讀 167,834評(píng)論 0 360
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)零抬。 經(jīng)常有香客問(wèn)我镊讼,道長(zhǎng),這世上最難降的妖魔是什么平夜? 我笑而不...
    開封第一講書人閱讀 59,543評(píng)論 1 296
  • 正文 為了忘掉前任蝶棋,我火速辦了婚禮,結(jié)果婚禮上忽妒,老公的妹妹穿的比我還像新娘玩裙。我一直安慰自己,他們只是感情好段直,可當(dāng)我...
    茶點(diǎn)故事閱讀 68,547評(píng)論 6 397
  • 文/花漫 我一把揭開白布吃溅。 她就那樣靜靜地躺著,像睡著了一般鸯檬。 火紅的嫁衣襯著肌膚如雪罕偎。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 52,196評(píng)論 1 308
  • 那天京闰,我揣著相機(jī)與錄音颜及,去河邊找鬼。 笑死蹂楣,一個(gè)胖子當(dāng)著我的面吹牛俏站,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播痊土,決...
    沈念sama閱讀 40,776評(píng)論 3 421
  • 文/蒼蘭香墨 我猛地睜開眼肄扎,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來(lái)了赁酝?” 一聲冷哼從身側(cè)響起犯祠,我...
    開封第一講書人閱讀 39,671評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎酌呆,沒想到半個(gè)月后衡载,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 46,221評(píng)論 1 320
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡隙袁,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 38,303評(píng)論 3 340
  • 正文 我和宋清朗相戀三年痰娱,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片菩收。...
    茶點(diǎn)故事閱讀 40,444評(píng)論 1 352
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡梨睁,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出娜饵,到底是詐尸還是另有隱情坡贺,我是刑警寧澤,帶...
    沈念sama閱讀 36,134評(píng)論 5 350
  • 正文 年R本政府宣布箱舞,位于F島的核電站遍坟,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏褐缠。R本人自食惡果不足惜政鼠,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,810評(píng)論 3 333
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望队魏。 院中可真熱鬧公般,春花似錦、人聲如沸胡桨。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,285評(píng)論 0 24
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)昧谊。三九已至刽虹,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間呢诬,已是汗流浹背涌哲。 一陣腳步聲響...
    開封第一講書人閱讀 33,399評(píng)論 1 272
  • 我被黑心中介騙來(lái)泰國(guó)打工胖缤, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人阀圾。 一個(gè)月前我還...
    沈念sama閱讀 48,837評(píng)論 3 376
  • 正文 我出身青樓哪廓,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親初烘。 傳聞我的和親對(duì)象是個(gè)殘疾皇子涡真,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,455評(píng)論 2 359

推薦閱讀更多精彩內(nèi)容