静态类型/静态分派/动态分派/单分派/多分派

静态类型/实际类型

Human woman = new Woman(); // Human就是静态类型,其实就是变量编译时类型,Woman就是实际类型
Man man = new Man(); // 静态类型和实际类型都是Man

方法解析

Class 文件的编译过程中不包含传统编译中的连接步骤,一切方法调用在 Class 文件里面存储的都只是符号引用,而不是方法在实际运行时内存布局中的入口地址。这个特性给 Java 带来了更强大的动态扩展能力,使得可以在类运行期间才能确定某些目标方法的直接引用,称为动态连接,也有一部分方法的符号引用在类加载阶段或第一次使用时转化为直接引用,这种转化称为静态解析。这在前面的“Java 内存区域与内存溢出”一文中有提到。

静态解析成立的前提是:方法在程序真正执行前就有一个可确定的调用版本,并且这个方法的调用版本在运行期是不可改变的。换句话说,调用目标在编译器进行编译时就必须确定下来,这类方法的调用称为解析。

在 Java 语言中,符合“编译器可知,运行期不可变”这个要求的方法主要有静态方法和私有方法两大类,前者与类型直接关联,后者在外部不可被访问,这两种方法都不可能通过继承或别的方式重写出其他的版本,因此它们都适合在类加载阶段进行解析。

Java 虚拟机里共提供了四个方法调用指令:

invokestatic:调用静态方法。
invokespecial:调用实例构造器方法、私有方法和父类方法。
invokevirtual:调用所有的虚方法。
invokeinterface:调用接口方法,会在运行时再确定一个实现此接口的对象。

只要能被 invokestatic 和 invokespecial 指令调用的方法,都可以在解析阶段确定唯一的调用版本,符合这个条件的有静态方法、私有方法、实例构造器和父类方法四类,它们在类加载时就会把符号引用解析为该方法的直接引用。这些方法可以称为非虚方法(还包括 final 方法),与之相反,其他方法就称为虚方法(final 方法除外)。这里要特别说明下 final 方法,虽然调用 final 方法使用的是 invokevirtual 指令,但是由于它无法覆盖,没有其他版本,所以也无需对方发接收者进行多态选择。Java 语言规范中明确说明了 final 方法是一种非虚方法。

解析调用一定是个静态过程,在编译期间就完全确定,在类加载的解析阶段就会把涉及的符号引用转化为可确定的直接引用,不会延迟到运行期再去完成。而分派调用则可能是静态的也可能是动态的,根据分派依据的宗量数(方法的调用者和方法的参数统称为方法的宗量)又可分为单分派和多分派。两类分派方式两两组合便构成了静态单分派、静态多分派、动态单分派、动态多分派四种分派情况。

静态分派

依赖静态类型来确定被调用的目标方法的分派过程称为静态分派。静态分派的典型应用就是方法重载。
调用重载的方法使用静态分派。

动态单分派

调用重写的方法使用动态分派
在运行期根据实际类型来确定被调用的目标方法的分派过程称为动态分派。

参考

1.https://www.zhihu.com/question/537174884
2.https://blog.csdn.net/sunxianghuang/article/details/52280002?utm_source=copy
3.https://wenku.baidu.com/view/a45d383451d380eb6294dd88d0d233d4b14e3f0c.html
4.https://blog.csdn.net/ns_code/article/details/17965867
5.http://www.javashuo.com/article/p-dwxxsokz-ky.html

猜你喜欢

转载自blog.csdn.net/liaowenxiong/article/details/125233405
今日推荐