JVM栈帧内部结构-动态链接
动态链接(或运行时常量池的方法引用):
每一个栈帧内部都包含一个指向运行时常量池中该栈帧所属方法的引用,包含这个引用的目的就是为了支持当前方法的代码能够实现动态链接(Dynamic Linking)。比如:invokedynamic指令
在Java源文件被编译到字节码文件时,所有的变量和方法引用都作为符号引用(Symbilic Reference)保存在class文件的常量池里。
比如:描述一个方法调用了另外的其他方法时,就是通过常量池中指向方法的符号引用来表示的,动态链接的作用就是为了将这些符号引用转换位调用方法的直接引用。
方法的调用:
在Jvm中,将符号引用转换位调用方法的直接引用,与方法的绑定机制相关。
静态链接:
当一个字节码文件被装载进JVM内部时,如果被调用的目标方法在编译期可知,且运行期保持不变。
将调用方法的符号引用转换为直接引用的过程称为静态链接。
动态链接:
被调用的目标方法在编译期无法被确定下来,只能够在程序运行期将方法的符号引用转换为直接引用,这种引用转换的过程具备动态性,称为动态链接。
方法的绑定机制分为早期绑定(Early Binding)和晚期绑定(Late Bingind)。绑定是一个字段、方法或类在符号引用被替换为直接引用的过程。
早期绑定:
被调用的目标方法在编译期可知,且运行保持不变。
晚期绑定:
被调用方法在编译期无法被确定下来,只能够在程序运行期根据实际类型绑定相关的方法。
虚方法与非虚方法:
如果方法在编译期就确定具体调用版本,这个版本在运行期间是不可变的。
静态方法、私有方法、final方法、实例构造器、父类方法都是非虚方法,其它方法都为虚方法。
虚拟机中方法调用指令:
普通调用指令:
1) invokestatic:调用静态方法,解析阶段确定唯一方法版本。
2) invokespecial:调用
3) invokevirtual:调用所有虚方法。
4) invokeinterface:调用接口方法。
动态调用指令:
5) invokedynamic:动态解析出需要调用的方法,然后执行
invokeinterface:固化在虚拟机内部,方法的调用执行不可人为干预。
invokedynamic:指令支持用户确定方法版本。
invokestatic:指令和invokespecial指令调用的方法称为非虚方法,其余(final修饰除外)称为虚方法。