Java 静态多分派&动态单分派
分派发生在编译期和运行期,编译期的分派为静态分派,运行期的为动态分派。编译期是根据对象声明的类型来选择方法,运行期是根据对象实际类型来选择方法。术语: 宗量(JVM虚拟机) , 什么是宗量, 方法调用者和方法参数被称为宗量.(后面理解分派需要)静态类型: 一个对象在声明时的类型称为静态类型,静态类型再编译器编译时可知. 如 Animal a = new Dog(), 静态...
分派发生在编译期和运行期,编译期的分派为静态分派,运行期的为动态分派。
编译期是根据对象声明的类型来选择方法,运行期是根据对象实际类型来选择方法。
术语: 宗量(JVM虚拟机) , 什么是宗量, 方法调用者和方法参数被称为宗量.(后面理解分派需要)
静态类型: 一个对象在声明时的类型称为静态类型,静态类型再编译器编译时可知. 如 Animal a = new Dog(), 静态类型为Animal, 实际类型为Dog.
Java 静态分派(方法重载)
public class Test{ //hi 方法重载 public void hi(Father f , Father f1){ System.out.println("ff"); } public void hi(Father f , Son s){ System.out.println("fs"); } public void hi(Son s , Son s2){ System.out.println("ss"); } public void hi(Son s , Father f){ System.out.println("sf"); } public static void main(String[] rags){ Father f = new Father(); Father s = new Son(); Test t = new Test(); t.hi(f , new Father()); t.hi(f , s); t.dost(s, f); } } class Father {} class Son extends Father{}
执行结果没有像预期的那样输出 ff、fs、sf而是输出了三个 ff.
此处对于对象声明时,静态类型为Father, 所以在编译期间,编译器会根据参数的静态类型选择要执行的方法,此时已经确定要执行的方法,所以在运行时调用的方法为ff输出的方法.这就是静态分派.
Java 动态分派(方法重写)
运行结果:f.i 0 s.i 0 WeclomeFather! WeclomeSon!public class Test{ public static void main(String[] rags){ Father f = new Father(); Father s = new Son(); System.out.println("f.i " +f.i); System.out.println("s.i " +s.i); f.hi(); s.hi(); } } class Father { int i = 0 ; public void hi(){ System.out.println("WelcomeFather!"); } } class Son extends Father{ int i = 9 ; public void hi(){ System.out.println("WelcomeSon!"); } }
变量f,s在编译器静态类型为Father,所以i来自于father, 在运行期间,JVM会根据实际类型来调用方法,s的实际类型为Son,所以调用的方法是Son重写的hi方法. 根据实际类型的方法调用为动态分派.
单分派&多分派
单分派和多分派取决于宗量, 方法调用者和方法参数都是宗量.
Java中静态分派的方法调用,首先确定调用者的静态类型是什么,然后根据要调用的方法参数的静态类型(声明类型)确定所有重载方法中要调用哪一个, 需要根据这两个宗量来编译, 所以是静态多分派(多个宗量确定).
Java中动态分派的方法调用,在运行期间,虚拟机会根据调用者的实际类型调用对应的方法, 只需根据这一个宗量就可以确定要调用的方法,所以是动态单分派(一个宗量)
以上是个人对于Java分派的理解, 如有偏差,请别打脸.
更多推荐
所有评论(0)