java Class文件結(jié)構(gòu)解析常量池字節(jié)碼
Class文件結(jié)構(gòu)
整體結(jié)構(gòu)
ClassFile {
u4 magic;
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];
}從結(jié)構(gòu)上看可以分為幾大塊 文件頭、常量池、接口、字段、函數(shù)、屬性
文件頭
u4 magic; //固定值 0xCAFEBABE
u2 minor_version;
u2 major_version;
u2 access_flags; //訪問修飾
u2 this_class; //類名 常量池下標(biāo)
u2 super_class; //父類名 常量池下標(biāo) 如果是0 就是java/lang/Object;
我把這幾個描述了類基本信息的字段稱為文件頭
major_version.minor_version表示該class文件的版本號,由編譯器版本決定,然而不同版本的虛擬機(jī)只會支持一定版本范圍內(nèi)的class文件,如果不在則會拒絕解析。
例如 openJDK中的實現(xiàn)
// Check version numbers - we check this even with verifier off
if (!is_supported_version(major_version, minor_version)) {
if (name == NULL) {
Exceptions::fthrow(
THREAD_AND_LOCATION,
vmSymbols::java_lang_UnsupportedClassVersionError(),
"Unsupported major.minor version %u.%u",
major_version,
minor_version);
} 常量池
常量池包含了class文件中使用到的例如 函數(shù)標(biāo)識/類型信息/字符串等等,運(yùn)行時加載到內(nèi)存中形成運(yùn)行時常量池
常量項中文件中使用變長結(jié)構(gòu)描述
cp_info {
u1 tag; //表示常量的類型
u1 info[]; //具體常量的內(nèi)容結(jié)構(gòu) 不同類型常量有不同的結(jié)構(gòu)
}
/*常量類型*/
Constant Type Value
CONSTANT_Class 7
CONSTANT_Fieldref 9
CONSTANT_Methodref 10
CONSTANT_InterfaceMethodref 11
CONSTANT_String 8
CONSTANT_Integer 3
CONSTANT_Float 4
CONSTANT_Long 5
CONSTANT_Double 6
CONSTANT_NameAndType 12
CONSTANT_Utf8 1
CONSTANT_MethodHandle 15
CONSTANT_MethodType 16
CONSTANT_InvokeDynamic 18例如:Utf8_info常量,更多的可以查看規(guī)范
CONSTANT_Utf8_info {
u1 tag;
u2 length; //字符串長度
u1 bytes[length]; //字符串?dāng)?shù)據(jù)(utf-8編碼)
}解析常量池的時候要注意:常量池長度為 constant_pool_count -1 但是 下標(biāo)從1開始
屬性表
attribute_info {
u2 attribute_name_index;
u4 attribute_length;
u1 info[attribute_length];
}屬性項可以包含在class、method、field、code中,作為具體信息項
在class中可以描述文件信息,在method中可以描述字節(jié)碼內(nèi)容,函數(shù)棧信息,在code中可以描述行號等 所以attribute_info同樣是具備不同類型的變長結(jié)構(gòu)。但是attribute_info并沒有tag這樣的專門標(biāo)記去標(biāo)識類型,而是使用名字attribute_name。
//一部分Attribute Name Attribute Section class file Java SE ConstantValue §4.7.2 45.3 1.0.2 Code §4.7.3 45.3 1.0.2 StackMapTable §4.7.4 50.0 6 Exceptions §4.7.5 45.3 1.0.2 InnerClasses §4.7.6 45.3 1.1 //.......
Code_attribute結(jié)構(gòu)
Code_attribute {
u2 attribute_name_index;
u4 attribute_length;
u2 max_stack; //最大棧深度
u2 max_locals; //局部變量數(shù)量
u4 code_length; //字節(jié)碼內(nèi)容長度
u1 code[code_length]; //字節(jié)碼內(nèi)容
u2 exception_table_length; //異常處理表 由try,catch/finaly 產(chǎn)生
{ u2 start_pc;
u2 end_pc;
u2 handler_pc;
u2 catch_type;
} exception_table[exception_table_length];
u2 attributes_count;
attribute_info attributes[attributes_count];
}函數(shù)表/字段表
method_info {
u2 access_flags;
u2 name_index;
u2 descriptor_index;
u2 attributes_count;
//一定會包含一個code屬性項
attribute_info attributes[attributes_count];
}
field_info {
u2 access_flags;
u2 name_index;
u2 descriptor_index;
u2 attributes_count;
attribute_info attributes[attributes_count];
}可以看到函數(shù)/字段中的內(nèi)容具體有屬性來描述
字節(jié)碼解析
對于class文件解析,我們最關(guān)心的可能就兩個 常量池和字節(jié)碼
一條字節(jié)碼由操作碼,操作數(shù)組成,不同的字節(jié)碼操作數(shù)的長度/表示意義可能不一樣,對著規(guī)范翻譯就好
例如invokevirtual字節(jié)碼就是 0xb6 u2 2字節(jié)的操作數(shù)在運(yùn)行時指向的是一個instance method ref
參考文檔
本文代碼 ClassParserDemo
以上就是java Class文件結(jié)構(gòu)解析常量池字節(jié)碼的詳細(xì)內(nèi)容,更多關(guān)于java Class 文件解析的資料請關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
SpringMVC+Mybatis實現(xiàn)的Mysql分頁數(shù)據(jù)查詢的示例
本篇文章主要介紹了SpringMVC+Mybatis實現(xiàn)的Mysql分頁數(shù)據(jù)查詢的示例,具有一定的參考價值,有興趣的可以了解一下2017-08-08
自從在 IDEA 中用了熱部署神器 JRebel 之后,開發(fā)效率提升了 10(真棒)
在javaweb開發(fā)過程中,使用熱部署神器 JRebel可以使class類還是更新spring配置文件都能立馬見到效率,本文給大家介紹JRebel的兩種安裝方法,小編建議使用第二種方法,具體安裝步驟跟隨小編一起看看吧2021-06-06
SpringBoot+Redis實現(xiàn)查找附近用戶的示例代碼
SpringDataRedis提供了十分簡單的地理位置定位的功能,本文主要介紹了SpringBoot+Redis實現(xiàn)查找附近用戶的示例代碼,具有一定的參考價值,感興趣的可以了解一下2024-02-02
Java實現(xiàn)讀取resources目錄下的文件路徑的九種方式
本文主要介紹了Java實現(xiàn)讀取resources目錄下的文件路徑的九種方式,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2022-04-04
Java下載文件中文文件名亂碼的解決方案(文件名包含很多%)
Java下載文件時,文件名中文亂碼問題通常是由于編碼不正確導(dǎo)致的,使用`URLEncoder.encode(filepath, "UTF-8")`可以解決在提示下載框中正確顯示漢字文件名的問題,但在選擇直接打開時,文件名會變成亂碼,解決這個問題的方法2025-02-02

