Class文件解析
Class文件结构 整体结构
ClassFile {
u4magic;
u2minor_version;
u2major_version;
u2constant_pool_count;
cp_infoconstant_pool[constant_pool_count-1];
u2access_flags;
u2this_class;
u2super_class;
u2interfaces_count;
u2interfaces[interfaces_count];
u2fields_count;
field_infofields[fields_count];
u2methods_count;
method_infomethods[methods_count];
u2attributes_count;
attribute_info attributes[attributes_count];
}
从结构上看可以分为几大块 文件头、常量池、接口、字段、函数、属性
文件头
u4magic;
//固定值 0xCAFEBABE
u2minor_version;
u2major_version;
u2access_flags;
//访问修饰
u2this_class;
//类名 常量池下标
u2super_class;
//父类名 常量池下标如果是0 就是java/lang/Object;
我把这几个描述了类基本信息的字段称为文件头major_version.minor_version表示该class文件的版本号,由编译器版本决定,然而不同版本的虚拟机只会支持一定版本范围内的class文件,如果不在则会拒绝解析。 例如 openJDK中的实现
// 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文件中使用到的例如 函数标识/类型信息/字符串等等,运行时加载到内存中形成运行时常量池
常量项中文件中使用变长结构描述
cp_info {
u1 tag;
//表示常量的类型
u1 info[];
//具体常量的内容结构 不同类型常量有不同的结构
}/*常量类型*/
Constant TypeValue
CONSTANT_Class7
CONSTANT_Fieldref9
CONSTANT_Methodref10
CONSTANT_InterfaceMethodref 11
CONSTANT_String 8
CONSTANT_Integer3
CONSTANT_Float4
CONSTANT_Long5
CONSTANT_Double 6
CONSTANT_NameAndType12
CONSTANT_Utf81
CONSTANT_MethodHandle15
CONSTANT_MethodType 16
CONSTANT_InvokeDynamic18
例如:Utf8_info常量,更多的可以查看规范
CONSTANT_Utf8_info {
u1 tag;
u2 length;
//字符串长度
u1 bytes[length];
//字符串数据(utf-8编码)
}
【Class文件解析】解析常量池的时候要注意:常量池长度为 constant_pool_count -1 但是 下标从1开始
属性表
attribute_info {
u2 attribute_name_index;
u4 attribute_length;
u1 info[attribute_length];
}
属性项可以包含在class、method、field、code中,作为具体信息项
在class中可以描述文件信息,在method中可以描述字节码内容,函数栈信息,在code中可以描述行号等 所以attribute_info同样是具备不同类型的变长结构。但是attribute_info并没有tag这样的专门标记去标识类型,而是使用名字attribute_name。
//一部分Attribute Name
AttributeSection class fileJava SE
ConstantValue§4.7.245.31.0.2
Code§4.7.345.31.0.2
StackMapTable§4.7.450.06
Exceptions§4.7.545.31.0.2
InnerClasses§4.7.645.31.1
//.......
Code_attribute结构
Code_attribute {
u2 attribute_name_index;
u4 attribute_length;
u2 max_stack;
//最大栈深度
u2 max_locals;
//局部变量数量
u4 code_length;
//字节码内容长度
u1 code[code_length];
//字节码内容
u2 exception_table_length;
//异常处理表 由try,catch/finaly 产生
{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];
}
函数表/字段表
method_info {
u2access_flags;
u2name_index;
u2descriptor_index;
u2attributes_count;
//一定会包含一个code属性项
attribute_info attributes[attributes_count];
}field_info {
u2access_flags;
u2name_index;
u2descriptor_index;
u2attributes_count;
attribute_info attributes[attributes_count];
}
可以看到函数/字段中的内容具体有属性来描述
字节码解析 对于class文件解析,我们最关心的可能就两个 常量池和字节码
一条字节码由操作码,操作数组成,不同的字节码操作数的长度/表示意义可能不一样,对着规范翻译就好
例如invokevirtual字节码就是 0xb6 u2 2字节的操作数在运行时指向的是一个instance method ref
参考文档
- JVM规范-Class文件结构
- JVM规范-字节码
推荐阅读
- django-前后端交互
- 如何在Mac中的文件选择框中打开系统隐藏文件夹
- 使用composer自动加载类文件
- ssh生成公钥秘钥
- Quartz|Quartz 源码解析(四) —— QuartzScheduler和Listener事件监听
- Java内存泄漏分析系列之二(jstack生成的Thread|Java内存泄漏分析系列之二:jstack生成的Thread Dump日志结构解析)
- [源码解析]|[源码解析] NVIDIA HugeCTR,GPU版本参数服务器---(3)
- Android系统启动之init.rc文件解析过程
- 小程序有哪些低成本获客手段——案例解析
- 微信小程序基础知识