多态是面向对象的核心思想之一,多态的实现有三要素: 1、 存在继承 2、子类对父类的方法进行了重写 3、父类引用指向子类对象。
前面说的还是有点虚,下面通过一个例子来深入理解多态。
程序代码如下,给出对应的输出:
1 public class Polymorphism { 2 public static void main(String[] args) { 3 A a1 = new A(); 4 A a2 = new B(); 5 B b = new B(); 6 C c = new C(); 7 D d = new D(); 8 9 a1.show(b); 10 a1.show(c); 11 a1.show(d); 12 a2.show(b); 13 a2.show(c); 14 a2.show(d); 15 b.show(b); 16 b.show(c); 17 b.show(d); 18 b.show(a1); 19 a1.show(a1); 20 a2.show(a1); 21 } 22 } 23 24 class A { 25 public void show(A a) { 26 System.out.println("A and A"); 27 } 28 29 public void show(D d) { 30 System.out.println("A and D"); 31 } 32 } 33 34 class B extends A { 35 public void show(B b) { 36 System.out.println("B and B"); 37 } 38 39 public void show(A a) { 40 System.out.println("B and A"); 41 } 42 } 43 44 class C extends B { 45 } 46 47 class D extends B { 48 }
下面我就来依次说说每一行的输出并给出为什么:
1. 对于a1.show(b) 因为a1的类型是A,所以只能调用A中的show(A a)方法 ----> 输出是 A and A
2. 对于a1.show(c) 理由同上 ------> 输出也是 A and A
3. 对于a1.show(d),此时a1的类型是A,但是此时因为传入的参数是D类型的,所以此时调用的是show(D d)方法 ----> 输出是A and D
4. 对于a2.show(b),此时a2是父类的引用指向子类的对象,虽然a2的实际类型是B,但是在方法调用上只能调用A的方法。所以此时调用的是A中的show(A a)方法,但是这时候又存在多态:A中的show(A a)在B中被重写了,所以此时实际调用的是B中的show(A a)方法。 -----> 输出是 B and A
5. 对于a2.show(c).理由同上 -----> 输出是 B and A
6. 对于a2.show(d).此时调用的是A中的show(D d)方法 ----->输出是 A and D
7. 对于b.show(b). 此时b是B类型对象,此时应该调用B中的show(B b)方法 ------> 输出是 B and B
8. 对于b.show(c). 理由同上 ------> 输出是 B and B
9. 对于b.show(d).此时B是B类型的对象,父类是可以调用子类中的方法,在方法调用的时候,会去找最匹配的方法,发现A中的show(D d)最匹配。 -----> 输出时 A and D
10. 对于b,show(a1). 此时最匹配的是B中的show(A a)方法 ------> 输出是 B and A
11. 对于a1.show(a1). 不用想直接调用的是A中的show(A a)方法 -----> 输出是 A and A
12. 对于a2.show(a1). 根据多态可得 ------> 输出是 B and A