Java多态
可以实现动态编译,类型实现可扩展性
public class Person {
}
public class Student extends Person{
}
import OOP.duotai.Person;
import OOP.duotai.Student;
public class Application {
public static void main(String[] args) {
//一个类new出来的对象的实际类型是确定的
//如new Person();new成的一定是Person类
//但类new的对象可以指向的引用类型就不一定确定
Student s1 = new Student();
Person s2 = new Student();
//即:父类的引用Person s2,指向子类Student的类型
//存在继承关系时才可
Object s3 = new Student();
}
}
一个类new出来的对象的实际类型是确定的,如new Person();
new成的一定是Person类。但类new的对象可以指向的引用类型不一定确定:即:父类的引用Person s2,指向子类Student的类型
(存在继承关系时才可)
public class Person {
public void run(){
System.out.println("Father");
}
}
public class Student extends Person{
}
public class Application {
public static void main(String[] args) {
Student s1 = new Student();
Person s2 = new Student();
s2.run();//s2为Person类(声明为最终的实际类,此时声明的是Person类,即s2前面声明的类),Person类中本身就有run方法,直接调用
s1.run();//s1为Student类,尽管Student类没有run方法,但Student类继承了父类Person的run方法,因此s1也可以run,此时方法为父类Person的run方法
}
}
此时输出:
Father
Father
若子类Student中对方法重写:
public class Person {
public void run(){
System.out.println("Father");
}
}
public class Student extends Person{
@Override
public void run() {
System.out.println("son");
}
}
public class Application {
public static void main(String[] args) {
Student s1 = new Student();
Person s2 = new Student();
s2.run();//s2是Person类,但指向的是Student子类,当子类重写了父类的方法后,会执行子类重写的方法
s1.run();//s1是Student类,当子类重写了父类的方法后,走重写后的方法,即son方法
Person s3 = new Person();//此时s3为纯粹的Person类,未指向子类,因此其方法仍是用父类原有的方法
s3.run();
}
}
此时输出:
son
son
Father
即:当父类的引用(Person s2)指向子类(new Student)时,若子类重写了父类的方法,那么该对象调用的是重写后子类中的方法,若未重写还是调用父类中的方法
当父类未指向子类时,若子类重写了父类的方法,对该对象方法的调用没有影响,仍是直接调用父类本来的方法。
public class Person {
public void run(){
System.out.println("Father");
}
}
public class Student extends Person{
@Override
public void run() {
System.out.println("son");
}
public void eat(){
System.out.println("eat");
}
}
public class Application {
public static void main(String[] args) {
Student s1 = new Student();
Person s2 = new Student();
s2.eat();//这样直接写此行会报错,因为s2为Person类,而Person类中本身没有eat方法,eat方法是Student子类中的方法
s1.eat();
}
}
即:对象能执行哪些方法,主要看生成对象时其左侧声明的类型,如Person s2 = new Student();中左侧的Person,Person作为父类本身无eat方法时,无法调用该方法,(与右侧关系不大)
即:子类可以调用的方法都是自己本身已有的或继承父类的
父类(声明)虽然可以指向子类,但不能调用子类独有的方法;且若子类对继承的方法进行了重写,父类指向子类的生成对象调用的也将是重写后的方法
上述例子若.eat处不报错,应对s2进行强制转换,将其从Person类转为Student类:
public class Application {
public static void main(String[] args) {
Student s1 = new Student();
Person s2 = new Student();
((Student) s2).eat();//强制转换
s1.eat();
}
}
多态的注意事项:
- 多态是方法的多态,属性没有多态
- 多态的前提:只存在于父类与子类之间,两类需要有父子联系,否则会出现类型转换异常:ClassCastException
- 多态存在的条件:继承关系;方法需要重写;父类引用指向子类对象:Father f1 = new Son();
- static方法属于类,不属于实例,因此不可重写
- final常量修饰的方法不可重写
- private修饰的方法不可重写
多态即:同一个方法(重写后)可以根据发送对象的不同(父类是否指向子类)而采用不同的方法
一个对象的实际类型是确定的,但可以指向对象的引用类型有很多(父类或其他关系)。
关键词:instanceof、类型转换(引用类型之间的转换)强制转换/自动转换