Java 多态之方法调用顺序

作者: zYoung_Tang | 来源:发表于2019-01-10 15:25 被阅读3次

本文例子取于<深入理解java多态性><java提高篇(四)-----理解java的三大特性之多态>,分析 Java 继承链方法调用优先级以及执行流程。

本文相当于是我对上面两篇文章的理解。

public class A {
    public String show(D obj) {
        return ("A and D");
    }

    public String show(A obj) {
        return ("A and A");
    } 
}

public class B extends A{
    public String show(B obj){
        return ("B and B");
    }
    
    public String show(A obj){
        return ("B and A");
    } 
}

public class C extends B{}

public class D extends B{}

public class Test {
    public static void main(String[] args) {
        A a1 = new A();
        A a2 = new B();
        B b = new B();
        C c = new C();
        D d = new D();
        
        System.out.println("(1)"+a1.show(b));  
        System.out.println("(2)"+a1.show(c));  
        System.out.println("(3)"+a1.show(d));   
        System.out.println("(4)"+a2.show(b));   
        System.out.println("(5)"+a2.show(c));   
        System.out.println("(6)"+a2.show(d));   
        System.out.println("(7)"+b.show(b));     
        System.out.println("(8)"+b.show(c));   
        System.out.println("(9)"+b.show(d)); 

    }
}

输出结果:

(1)A and A
(2)A and A
(3)A and D
(4)B and A
(5)B and A
(6)A and D
(7)B and B
(8)B and B
(9)A and D

根据 Java 多态机制,继承链中对象方法的调用存在一个优先级:

this.method(O) -> super.method(O) -> this.method((super)O) -> super.method((super)O)

上面两篇文章中都提到多态机制遵循的规则可以概括为当超类对象引用变量引用子类对象时,被引用对象的类型而不是引用变量的类型决定了调用谁的成员方法,但是这个被调用的方法必须是在超类中定义过的。

以上面的例子举例,A a = new B(); 就是超类对象引用变量 a 引用了子类 B 的对象,最终由 B 来决定调用谁的成员方法,但前提是这个方法即 show 方法必须在超类中定义过的。

在这里我把这两句话重新排序并解读:

  • 当超类对象引用变量引用子类对象时,首先这个被调用的方法必须在超类中定义过 = 根据上面继承链对象方法调用优先级可以在超类中找到这个方法
  • 被引用对象的类型而不是引用变量的类型决定了调用谁的成员方法 = 判断子类是否重写了该方法,如果重写了直接调用子类的方法

简单来说就是先根据优先级确定目标方法,然后根据多态机制遵循的规则来决定调用子类还是超类的成员方法。


下面开始分析例子中的九种情况。

(1)(2)(3) 的例子都不是超类对象引用变量引用子类对象的情况,所以在超类中找到方法后直接调用即可。

(1) a1.show(b)

  • 根据优先级第一级 this.method(O) 可改成 A.show(B) ,即要在 A 中 寻找方法 show(B),由于 A 中没有 show(B) 的方法,所以进入第二级 super.method(O) 。
  • 因为 A 除了 Object 之外没有父类且 Object 中肯定没有 show 方法,所以进入第三级 this.method((super)O) 。
  • B 的父类是 A 所以目标是找 A.show(A) ,然后 A 中找到了 show(A) 方法并直接调用,返回 " A and A "。

(2) a1.show(c)

  • 进入第三级 this.method((super)O) 之前跟 (1) 一样。
  • C 的父类是 B,并没有在 A 中找到 show(B) 方法,所以会根据 C 的继承链继续向上找 B 的父类 A ,然后寻找 A.show(A) ,所以最终结果跟 (1) 一样是 " A and A "。

(3) a1.show(d)

  • 直接根据第一级即可找到目标方法 A.show(D) 并执行得到 " A and D "。

(4)(5)(6) 都是超类对象引用变量引用子类对象的情况,所以在超类中找到方法后还要根据多态机制规则决定执行子类还是超类的方法。

(4) a2.show(b)

  • 根据优先级第一级在 A 中找不到 show(B) ,所以进入第二级。
  • 因为 A 除了 Object 之外没有父类且 Object 中肯定没有 show 方法,所以进入第三级 。
  • 根据第三级在 A 中找到了 show(A) ,然后根据多态机制规则由于子类 B 重写了 show(A) 方法,所以最终调用的是 B.show(a) ,得到结果是 " B and A "。

(5) a2.show(c)

  • 进入第三级之前跟 (4) 一样。
  • 根据第三级在 A 中没有找到 show(B) ,所以会根据 B 的继承链向上找 B 的父类 A ,并在 A 中找到了 show(A) ,然后根据多态机制规则由于子类 B 重写了 show(A) 方法,所以最终调用的是 B.show(a) ,得到结果是 " B and A "。

(6)a2.show(d)

  • 直接根据第一级即可找到目标方法 A.show(D) ,然后根据多态机制规则由于子类 B 没有重写 show(A) 方法,所以执行超类方法 A.show(D) ,得到结果为 " A and D "。

至于(7)(8)(9)根据上面就可以推导出来。

相关文章

  • Java 多态之方法调用顺序

    本文例子取于<深入理解java多态性>和,分析 Ja...

  • Java 多态

    1、Java 多态分类 1)编译时多态:方法重载。2)运行时多态:JAVA运行时系统根据调用该方法的实例的类型来决...

  • java多态面试题

    java多态性 多态分两种: (1) 编译时多态(设计时多态):方法重载。 (2) 运行时多态:JAVA运...

  • JAVA_多态的方法调用

    其实在继承链中对象方法的调用存在一个优先级:this.show(O)、super.show(O)、this.sho...

  • java中 多态 父子类的调用顺序

    关于多态,可以有个结论,静态方法(fu.show())和成员变量看左边 成员方法 编译看左边 运行看右边,...

  • java多态面试题

    java多态性 多态分两种: (1) 编译时多态(设计时多态):方法重载。 (2) 运行时多态:JAVA运行时...

  • java编程思想读书笔记一

    多态 任何域的访问操作都将有编译器解析,如果某个方法是静态的,它的行为就不具有多态性 java默认对象的销毁顺序与...

  • Java多态相关

    多态的三个条件: 1、要有继承 2、要有重写 3、父类引用指向子类对象[Java多态之方法的重写](http://...

  • Java几个基本概念

    多态:通过父类指针调用子类方法 重载:一个类中方法参数名、个数、类型不同 重写:子类对父类方法的重写 Java泛型...

  • Java RMI之HelloWorld篇

    Java RMI之HelloWorld篇 Java RMI 指的是远程方法调用 (Remote Method In...

网友评论

    本文标题:Java 多态之方法调用顺序

    本文链接:https://www.haomeiwen.com/subject/qesarqtx.html