解析java中的多态

Java
292
0
0
2023-12-09

怎么使用多态

3.1 使用 多态 的前提

两个类之间 必须存在父子关系 (继承关系)

必须实现了方法的重写,因为多态主要是指的是 方法的多态性 ,和属性没关系

3.2 多态运行的机制

3.2.1 编译期(静态)

编译时看的是左边,编译阶段会检查对象.属性/方法是否是父类里面定义的属性和方法(若父类还有其父亲,也要在父类型特征区去进行寻找),若存在,才允许编译通过,若不存在则编译不通过

编译不通过示例代码

Animal01

 package Work;

public class Animal {
 
     private  String name;
    //姓名
    private int age;
    //年龄
    private String sex;
    //性别

    public Animal() {
 
    }

    public Animal(String name, int age,  String  sex) {
 
        this.name = name;
        this.age = age;
        this.sex = sex;
    }

    public String getName() {
 
        return name;
    }

    public  void  setName(String name) {
 
        this.name = name;
    }

    public int getAge() {
 
        return age;
    }

    public void setAge(int age) {
 
        this.age = age;
    }

    public String getSex() {
 
        return sex;
    }

    public void setSex(String sex) {
 
        this.sex = sex;
    }
    //使用多态后,父类只需要一个方法就可以使用所有动物对象不同的叫声的功能
    public void say(Animal animal01){
 
        System.out.println("名字为: "+animal.getName()+"年龄为: "+animal01.getAge()+"性别为: "+animal01.getSex()+"的狗在汪汪汪");
    }

} 

Dog01

 package Work;
public class Dog extends Animal01{
 
    int weight;
    //Dog独有的属性
    public Dog() {
 
    }

    public Dog(String name, int age, String sex) {
 
        super(name, age, sex);
    }
} 

Test

 package Work;
public class Test {
 
    public  static  void main(String[] args) {
 
        //建立一个父类对象,启动say方法
       Animal a=new Dog01("王富贵",5,"雌性");
       System.out.println(a.weight);;
    }
} 

错误示例截图

3.2.1 运行期(动态)

因为编译期会把 人类的独有方法被过滤掉 ,因此运行的时候只需要考虑.的方法是否在子类型 实例化的空间里面 存在了。

运行看的是右边,若子类 重写了.调用的方法 ,那么用多态时候调用的就是子类的方法,若没有重写,调用的是子类中的父类型特征区里面的方法(继承时从父类那边拷贝一份过来的)

示例代码

Animal01类

 package Work;

public class Animal {
 
    private String name;
    //姓名
    private int age;
    //年龄
    private String sex;
    //性别

    public Animal() {
 
    }

    public Animal(String name, int age, String sex) {
 
        this.name = name;
        this.age = age;
        this.sex = sex;
    }

    public String getName() {
 
        return name;
    }

    public void setName(String name) {
 
        this.name = name;
    }

    public int getAge() {
 
        return age;
    }

    public void setAge(int age) {
 
        this.age = age;
    }

    public String getSex() {
 
        return sex;
    }

    public void setSex(String sex) {
 
        this.sex = sex;
    }
    public void say(){
 
        System.out.println("动物在说话!!!");
    }
    public void test(){
 
        System.out.println("测试父类中没有被子类重写的方法是否能被调用");
    }

} 

Dog01类

 package Work;
public class Dog extends Animal01{
 
    int weight;
    //Dog独有的属性
    public Dog() {
 
    }

    public Dog(String name, int age, String sex) {
 
        super(name, age, sex);
    }

    @Override
    public void say() {
 
        System.out.println("狗狗在说话");
    }
} 
 package Work;
public class Test {
 
    public static void main(String[] args) {
 
        //建立一个父类对象,启动say方法
       Animal a=new Dog01("王富贵",5,"雌性");
       a.test();
       a.say();
    }
} 

示例代码运行截图

3.2.3 为什么多态运行的机制要设计成这样???

编译期间

在多个类之间进行参数的传递时,传递的具体是哪个类型,编译器并不清楚。在new的过程中,是哪个类型也不知道, 为了保证一定有一个方法能够在运行期间能够顺利启动 ,所以编译期才必须要求调用的方法一定是父类型的方法,调用的属性一定是父类型的属性

运行期间

为了体现 子类实例化对象的差异性 ,例如狗的叫声是汪汪汪,猫的叫声是喵喵喵

3.3 多态是编译时状态还是运行时状态

解答: 运行时状态 ,因为你在编译的过程中并不知道你 实例化 的对象是什么类型的,只有在方法调用的时候才知道

3.4 方法重载 是运行时状态还是编译时状态

解答:方法的重载是编译时状态,因为你在用方法的时候就确定是那个方法了,不会有其他的可能性的。是静态的

3.5 多态的三个常见的使用场景

a 父类中有一个方法且该方法没有被子类重写,不同的子类调用该方法时,想根据子类的类型去得到对应的结果

典型题目

存在一个父类Animal01,子类Dog01、子类Cat01,其中Animal类中有一个say()叫声方法,此方法没有被子类重写,然后你想要Dog01调用say()方法打印汪汪汪,Cat01调用时say()方法时打印喵喵喵

示例代码

Animal01类

 package Work;

public class Animal {
 
    private String name;
    //姓名
    private int age;
    //年龄
    private String sex;
    //性别

    public Animal() {
 
    }

    public Animal(String name, int age, String sex) {
 
        this.name = name;
        this.age = age;
        this.sex = sex;
    }

    public String getName() {
 
        return name;
    }

    public void setName(String name) {
 
        this.name = name;
    }

    public int getAge() {
 
        return age;
    }

    public void setAge(int age) {
 
        this.age = age;
    }

    public String getSex() {
 
        return sex;
    }

    public void setSex(String sex) {
 
        this.sex = sex;
    }
    public void say(Animal animal01){
 
       if(animal  instanceof  Dog01){
 
           System.out.println("汪汪汪");
       }else if(animal instanceof Cat01){
 
           System.out.println("喵喵喵");
       }
    }


} 

Cat01类

 package Work;

public class Cat extends  Animal01{
 
    public Cat() {
 
    }

    public Cat(String name, int age, String sex) {
 
        super(name, age, sex);
    }
} 

Dog01类

 package Work;
public class Dog extends Animal01{
 
    public Dog() {
 
    }

    public Dog(String name, int age, String sex) {
 
        super(name, age, sex);
    }

} 
 package Work;
public class Test {
 
    public static void main(String[] args) {
 
        //建立一个父类对象animal,启动say方法
        Animal animal01=new Animal01();
        animal.say(new Dog01());
        animal.say(new Cat01());
    }
} 

示例代码运行截图

b 父类中有一个方法且该方法没有被子类重写,不同的子类调用该方法时会去执行其独有的方法

典型题目

存在一个父类Animal01,子类Dog01、子类Cat01,其中Animal类中有一个action()行为方法,此方法没有被子类重写,然后你想要Dog01调用action()方法会调用Dog01中特有的方法guardDoor看门,Cat01调用时action()方法时会去调用Cat01中特有的方法catchMouse()

示例代码

Animal01类

 package Work;

public class Animal {
 
    private String name;
    //姓名
    private int age;
    //年龄
    private String sex;
    //性别

    public Animal() {
 
    }

    public Animal(String name, int age, String sex) {
 
        this.name = name;
        this.age = age;
        this.sex = sex;
    }

    public String getName() {
 
        return name;
    }

    public void setName(String name) {
 
        this.name = name;
    }

    public int getAge() {
 
        return age;
    }

    public void setAge(int age) {
 
        this.age = age;
    }

    public String getSex() {
 
        return sex;
    }

    public void setSex(String sex) {
 
        this.sex = sex;
    }
    public void action(Animal animal01){
 
       if(animal instanceof Dog01){
 
           ((Dog)animal01).guardDoor();
       }else if(animal instanceof Cat01){
 
           ((Cat)animal01).catchMouse();
       }
    }


} 

Cat01类

 package Work;

public class Cat extends  Animal01{
 
    public Cat() {
 
    }

    public Cat(String name, int age, String sex) {
 
        super(name, age, sex);
    }
    public void catchMouse(){
 
        System.out.println("猫咪在捉老鼠");
    }
} 

Dog01类

 package Work;
public class Dog extends Animal01{
 
    public Dog() {
 
    }

    public Dog(String name, int age, String sex) {
 
        super(name, age, sex);
    }
    public void guardDoor(){
 
        System.out.println("狗狗在看门");
    }
} 
 package Work;
public class Test {
 
    public static void main(String[] args) {
 
        //建立一个父类对象animal,启动action方法
        Animal animal01=new Animal01();
        animal.action(new Dog01());
        animal.action(new Cat01());
    }
} 

示例代码运行截图

c 父类中有一个方法且该方法被子类重写,不同的子类调用该方法时,调用的是子类中重写的方法,从而表现出子类对象的差异性

典型题目

存在一个父类Animal01,子类Dog01、子类Cat01,其中Animal类中有一个eat()叫声方法, 此方法被其所有子类重写 ,存在一个Master主人类,主人根据不同的实例化对象,进行不同的喂养方法

示例代码

Animal01类

 package Work;

public class Animal {
 
    private String name;
    //姓名
    private int age;
    //年龄
    private String sex;
    //性别

    public Animal() {
 
    }

    public Animal(String name, int age, String sex) {
 
        this.name = name;
        this.age = age;
        this.sex = sex;
    }

    public String getName() {
 
        return name;
    }

    public void setName(String name) {
 
        this.name = name;
    }

    public int getAge() {
 
        return age;
    }

    public void setAge(int age) {
 
        this.age = age;
    }

    public String getSex() {
 
        return sex;
    }

    public void setSex(String sex) {
 
        this.sex = sex;
    }
    public void eat(){
 
        System.out.println("父类对象提供方法给子类对象去重写它");
    }


} 

Cat01类

 package Work;

public class Cat extends  Animal01{
 
    public Cat() {
 
    }

    public Cat(String name, int age, String sex) {
 
        super(name, age, sex);
    }
    @Override
    public void eat() {
 
        System.out.println("猫咪在吃鱼");
    }
} 

Dog01类

 package Work;
public class Dog extends Animal01{
 
    public Dog() {
 
    }

    public Dog(String name, int age, String sex) {
 
        super(name, age, sex);
    }

    @Override
    public void eat() {
 
        System.out.println("狗在啃骨头");
    }
} 

Master类

 package Work;
public class Master {
 
    public void feed(Animal animal01){
 
        animal.eat();
        //不用去进行对象是哪个类型的判断,调用是子类被重写的方法
        //结论:只要调用的是子类的重写的方法,那么就无需进行对象的类型的判断
    }
} 
 package Work;
public class Test {
 
    public static void main(String[] args) {
 
        Master master=new Master();
        master.feed(new Dog());
        master.feed(new Cat());
    }
} 

示例代码运行截图