1.首先需要熟记多态的首要条件:
1.1 有继承或实现接口
1.2 有方法重写
1.3 父类引用指向子类对象
2.理解向上转型和向下转型
向上转型:class A extends B{},B为A的父类,在实例化对象的时候,B test = new A();
这里,父类类型的引用test保存的是指向A类对象的实例的地址。称为父类引用指向子类对象;换句话说,子类对象使用的是父类类型的引用(表述不准确),表示向上转型了。
注意:向上转型之后,对象会遗失和父类不同的方法。即,向上转型之后的对象只能调用与父类方法名称相同的方法,不能调用父类不存在的方法。
向下转型:
package a.b;
public class A {
void aMthod() {
System.out.println("A method");
}
}
A的子类B:
package a.b;
public class B extends A {
void bMethod1() {
System.out.println("B method 1");
}
void bMethod2() {
System.out.println("B method 2");
}
}
C类:
package a.b;
public class C {
public static void main(String[] args) {
A a1 = new B(); // 向上转型
a1.aMthod(); // 调用父类aMthod(),a1遗失B类方法bMethod1()、bMethod2()
B b1 = (B) a1; // 向下转型,编译无错误,运行时无错误
b1.aMthod(); // 调用父类A方法
b1.bMethod1(); // 调用B类方法
b1.bMethod2(); // 调用B类方法
A a2 = new A();
B b2 = (B) a2; // 向下转型,编译无错误,运行时将出错
b2.aMthod();
b2.bMethod1();
b2.bMethod2();
}
}
为什么前一句向下转型代码可以,而后一句代码却出错?这是因为a1指向一个子类B的对象,所以子类B的实例对象b1当然也可以指向a1。而a2是一个父类对象,子类对象b2不能指向父类对象a2。
另外多态也可以体现在父类实例作为方法参数