一、多态的定义:
父类类型的变量指向子类创建的对象(eg:Father son = new Son();),使用该变量调用父类中一个被子类重写的方法(eg:son.hobby();),则父类中的该方法呈现出不同的行为特征(eg:下面代码中Son类和Daughter类分别重写了Father中的hobby()方法,在用子类实例化出来的对象调用hobby()时,hooby()呈现出了”儿子喜欢打篮球“和”女儿喜欢跳舞“两种特征),这就是多态。
public class Father {
int age;
int id;
String gender;
public void hobby() {
System.out.println("父亲喜欢看足球赛。");
}
}
public class Son extends Father{
int age;
int id;
String gender;
public void hobby() {
System.out.println("儿子喜欢打篮球。");
}
}
public class Daughter extends Father{
int age;
int id;
String gender;
public void hobby() {
System.out.println("女儿喜欢跳舞。");
}
}
public class Main {
public static void main(String[] args) {
Father son = new Son();
son.hobby();//运行结果为:儿子喜欢打蓝球。
Father dau = new Daughter();
dau.hobby();//运行结果为:女儿喜欢跳舞。
}
}
二、具体分析
Java引用变量有两种类型,分别是编译时类型和运行时类型:编译时类型由声明该变量时使用的类型决定;运行时类型由实际赋给该变量的对象。如果编译时类型和运行时类型不一致,就可能出现所谓多态。
上例分析:当把子类创建的对象直接赋给父类引用类型时,例如上例Main方法中“Father son = new Son();”, son引用变量的编译时类型是Father,运行时类型是Son,当程序运行时,该引用变量son调用父类中被子类重写的方法时,其方法行为表现的是子类重写该方法后的行为特征,而不是父类方法的行为特征。
注:如果编译时类型和运行时类型不一致时未必会出现所谓多态,如下例子:
public class Father {
int age;
int id;
String gender;
public void hobby() {
System.out.println("父亲喜欢看足球赛。");
}
}
public class Son extends Father{
int age;
int id;
String gender;
}
public class Main {
public static void main(String[] args) {
Father son = new Son();
son.hobby();//因Son类中并未重写hobby()方法,所以运行结果为:父亲喜欢看足球赛。
}
}
三、补充
1、上转型对象:子类实例化的对象赋值给父类声明变量,则该对象称为上转型对象,这个过程称为对象上转型,对应于数据类型转换中的自动类型转换:
public class Main {
public static void main(String[] args) {
Father son = new Son();//此处为对象上转型
}
}
注:①.上转型对象不能操作子类新增的成员变量;不能调用子类新增的方法。
②.上转对象调用父类方法,如果该方法已被子类重写,则表现子类重写后的行为特征,否则表现 父类的行为特征。
③. 使用上转型对象调用成员变量,无论该成员变量是否已经被子类覆盖,使用的都是父类中的成员变量:
2、对象下转型:
①可以将上转型对象再强制转换为创建该对象的子类类型的对象,即将上转型对象还原为子类对象,对应于数据类型转换中的强制类型转换。
②还原后的对象又具备了子类所有属性和功能,即可以操作子类中继承或新增的成员变量,可以调用子类中继承或新增的方法。
注意:不可以将父类创建的对象通过强制类型转换赋值给子类声明的变量。