1.首先我们先看一段代码
fulei:
package sirius;
public class Mammal {
public void move() {
System.out.println("正在移动......");
}
}
zilei:
package sirius;
public class Whale extends Mammal {
public void move() {
System.out.println("靠鳍移动......");
}
}
package sirius;
public class Earth {
public static void main(String[] args) {
Whale whale = new Whale();
whale.move();
上面这段代码包括了父类,子类和执行方法的代码,在这里,需要首先指出的是当子类对象调用重写后的方法时,调用的是子类重写后的方法,即上述执行代码后的运行结果是“靠鳍移动......”
2.接下来引入自动类型转换的概念
//double price = 9;
Mammal mammal = new Whale();
自动类型转换:父类类型的变量指向其子类创建的对象 对象上转型 new Whale()上转型对象
此时该自动类型转换可称为一个对象上转型过程,这时的new Whale()即为一个上转型对象
3.由代码运行引出多态
package sirius;
public class Earth {
public static void main(String[] args) {
Whale whale = new Whale();
whale.move();
//double price = 9;
Mammal mammal = new Whale();
mammal.move();
如果按着ctrl键,再点击mammal.move();的move时,会指向父类中的mammal,但是此时如果执行该代码,则会得到子类中重写方法的结果。该过程则可以理解为拥有两个状态:表面调用的是父类的方法,但是执行时由于mammal存的是子类创建的对象,所以执行的是子类中的方法。这就是Java里的多态。
4.什么是多态
父类类型(比如Mammal)的变量(比如mammal1)指向子类创建的对象,使用该变量调用父类中一个被子类重写的方法(比如move方法),则父类中的方法呈现出不同的行为特征,这就是多态。 Java引用变量有两种类型,分别是编译时类型和运行时类型:编译时类型由声明该变量时使用的类型决定;运行时类型由实际赋给该变量的对象。如果编译时类型和运行时类型不一致,就可能出现所谓多态。
5.在多态中需要注意的一些问题
如果编译时类型和运行时类型相同,则一定不会出现多态;√
如果编译时类型与运行时类型不相同,则一定出现多态(×);多态需要子类重写父类的方法
6.上转型对象和下转型对象
上转型对象不能调用子类新增加的方法和新增的属性,若想在上转型对象中实现调用子类新增加的方法和新增加的属性,则需要引入对象下转型的概念,下面我们先看一段代码:
//int price = (int)9.0
Whale whale = (Whale)mammal;
whale.
此时whale.后面可以是子类中新增的方法和属性。但是需要注意的是,下转型的前提是先出现上转型对象,像下面一段代码,因为没有出现上转型对象就用了下转型,则whale.后面就不会出现子类中新增的方法和属性。
Mammal mammal = new Mammal();
Whale whale = (Whale)mammal;
whale.
上转对象调用父类方法,如果该方法已被子类重写,则表现子类重写后的行为特征,否则表现父类的行为特征。 使用上转型对象调用成员变量,无论该成员变量是否已经被子类覆盖,使用的都是父类中的成员变量。
可以将上转型对象再强制转换为创建该对象的子类类型的对象,即将上转型对象还原为子类对象,对应于数据类型转换中的强制类型转换。 还原后的对象又具备了子类所有属性和功能,即可以操作子类中继承或新增的成员变量,可以调用子类中继承或新增的方法。 注意:不可以将父类创建的对象通过强制类型转换赋值给子类声明的变量。
7.抽象类方法体
在多态的前提下,父类中被子类重写的方法没有必要有方法体
此时如果父类的方法中有一个抽象方法的话,则该父类为一个抽象方法类;若该父类中的方法全都为抽象方法,则该父类就成为一个接口。