Java多态

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();
    }
}

多态的注意事项:

  1. 多态是方法的多态,属性没有多态
  2. 多态的前提:只存在于父类与子类之间,两类需要有父子联系,否则会出现类型转换异常:ClassCastException
  3. 多态存在的条件:继承关系;方法需要重写;父类引用指向子类对象:Father f1 = new Son();
    1. static方法属于类,不属于实例,因此不可重写
    2. final常量修饰的方法不可重写
    3. private修饰的方法不可重写

多态即:同一个方法(重写后)可以根据发送对象的不同(父类是否指向子类)而采用不同的方法

一个对象的实际类型是确定的,但可以指向对象的引用类型有很多(父类或其他关系)。

关键词:instanceof、类型转换(引用类型之间的转换)强制转换/自动转换

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值