标签:sub pad pack object类 查找 obj 指针 描述 同时存在
解析阶段是虚拟机将常量池内的符号引用替换为直接引用的过程,符号引用在前一章讲解
Class文件格式的时候已经出现过多次,在Class文件中它以CONSTANT_Class_info、
CONSTANT_Fieldref_info、CONSTANT_Methodref_info等类型的常量出现,那解析阶段中所说的直接引用与符号引用又有什么关联呢?
符号引用(Symbolic References):符号引用以一组符号来描述所引用的目标,符号可以是任何形式的字面量,只要使用时能无歧义地定位到目标即可。符号引用与虚拟机实现的内存布局无关,引用的目标并不一定已经加载到内存中。各种虚拟机实现的内存布局可以各不相同,但是它们能接受的符号引用必须都是一致的,因为符号引用的字面量形式明确定义在
Java虚拟机规范的Class文件格式中。
直接引用(Direct References):直接引用可以是直接指向目标的指针、相对偏移量或是一个能间接定位到目标的句柄。直接引用是和虚拟机实现的内存布局相关的,同一个符号引用在不同虚拟机实例上翻译出来的直接引用一般不会相同。如果有了直接引用,那引用的目标必定已经在内存中存在。
虚拟机规范之中并未规定解析阶段发生的具体时间,只要求了在执行anewarray、
checkcast、getfield、getstatic、instanceof、invokedynamic、invokeinterface、
invokespecial、invokestatic、invokevirtual、ldc、ldc_w、multianewarray、new、
putfield和putstatic这16个用于操作符号引用的字节码指令之前,先对它们所使用的符号引用进行解析。所以虚拟机实现可以根据需要来判断到底是在类被加载器加载时就对常量池中的符号引用进行解析,还是等到一个符号引用将要被使用前才去解析它。对同一个符号引用进行多次解析请求是很常见的事情,除invokedynamic指令以外,虚拟机实现可以对第一次解析的结果进行缓存(在运行时常量池中记录直接引用,并把常量标识为已解析状态)从而避免解析动作重复进行。无论是否真正执行了多次解析动作,虚拟机需要保证的是在同一个实体中,如果一个符号引用之前已经被成功解析过,那么后续的引用解析请求就应当一直成功;同样的,如果第一次解析失败了,那么其他指令对这个符号的解析请求也应该收到相同的异常。
对于invokedynamic指令,上面规则则不成立。当碰到某个前面已经由invokedynamic指令触发过解析的符号引用时,并不意味着这个解析结果对于其他invokedynamic指令也同样生效。因为invokedynamic指令的目的本来就是用于动态语言支持(目前仅使用Java语
言不会生成这条字节码指令),它所对应的引用称为"动态调用点限定符"(Dynamic
Call SiteSpecifier),这里"动态"的含义就是必须等到程序实际运行到这条指令的时候,解析动作才能进行。相对的,其余可触发解析的指令都是"静态"的,可以在刚刚完成加载阶段,还没有开始执行代码时就进行解析。
解析动作主要针对类或接口、字段、类方法、接口方法、方法类型、方法句柄和调用点限定符7类符号引用进行,分别对应于常量池的CONSTANT_Class_info、
CONSTANT_Fieldref_info、CONSTANT_Methodref_info、
CONSTANT_InterfaceMethodref_info、CONSTANT_MethodType_info、
CONSTANT_MethodHandle_info和CONSTANT_InvokeDynamic_info 7种常量类型
[1]。下面将讲解前面4种引用的解析过程,对于后面3种,与JDK 1.7新增的动态语言支持息
息相关,由于Java语言是一门静态类型语言,因此在没有介绍invokedynamic指令的语义之前,没有办法将它们和现在的Java语言对应上,笔者将在第8章介绍动态语言调用时一起分析讲解。
1.类或接口的解析假设当前代码所处的类为D,如果要把一个从未解析过的符号引用N解析为一个类或接口C 的直接引用,那虚拟机完成整个解析的过程需要以下3个步骤:
似"[Ljava/lang/Integer"的形式,那将会按照第1点的规则加载数组元素类型。如果N的描述符如前面所假设的形式,需要加载的元素类型就是"java.lang.Integer",接着由虚拟机生成一个代表此数组维度和元素的数组对象。
要解析一个未被解析过的字段符号引用,首先将会对字段表内class_index[2]项中索引的
CONSTANT_Class_info符号引用进行解析,也就是字段所属的类或接口的符号引用。如果在解析这个类或接口符号引用的过程中出现了任何异常,都会导致字段符号引用解析的失败。如果解析成功完成,那将这个字段所属的类或接口用C表示,虚拟机规范要求按照如下步骤对C进行后续字段的搜索。
在实际应用中,虚拟机的编译器实现可能会比上述规范要求得更加严格一些,如果有一个同名字段同时出现在C的接口和父类中,或者同时在自己或父类的多个接口中出现,那编译器将可能拒绝编译。在代码清单7-4中,如果注释了Sub类中的"public static int
A=4;",接口与父类同时存在字段A,那编译器将提示"The field Sub.A is ambiguous",不明确,并且拒绝编译这段代码。
代码清单
package demo.jvm.test6? public class FieldResolution { interface Interface0 { int A = 0? }
interface Interface1 extends Interface0 { int A = 1? }
interface Interface2 { int A = 2? }
static class Parent implements Interface1 { public static int A = 3? }
static class Sub extends Parent implements Interface2 { //public static int A = 4? }
public static void main(String[] args) { System.out.println(Sub.A)? } } |
类方法解析的第一个步骤与字段解析一样,也需要先解析出类方法表的class_index[3]项中索引的方法所属的类或接口的符号引用,如果解析成功,我们依然用C表示这个类,接下来虚拟机将会按照如下步骤进行后续的类方法搜索。
java.lang.IncompatibleClassChangeError异常。
接口方法也需要先解析出接口方法表的class_index[4]项中索引的方法所属的类或接口的符号引用,如果解析成功,依然用C表示这个接口,接下来虚拟机将会按照如下步骤进行后续的接口方法搜索。
由于接口中的所有方法默认都是public的,所以不存在访问权限的问题,因此接口方法的符号解析应当不会抛java.lang.IllegalAccessError异常。
从上面我们可以看到结果是解析就是我们平常在eclipse中看到的编译的过程,假如编译有问题就会报红了。
Jvm(55),虚拟机类加载机制----类加载的过程----解析
标签:sub pad pack object类 查找 obj 指针 描述 同时存在
原文地址:https://www.cnblogs.com/qingruihappy/p/9691476.html