Java 多态

一. 父类与子类之间的类型转换

public class Main {

    public static void main(String[] args) {
        /*
         * 向上转型/自动类型转换:将子类对象转换为父类类型
         *
         * 父类对象可以访问父类的属性和方法,并且方法是子类重写后的
         * 父类对象无法访问子类特有的属性和方法(不包括重写后的)
         */
        父类名 父类对象名 = new 子类名(实参);

        /*
         * 向下转型/强制类型转换:向上转型后,将父类对象转换为子类类型
         *
         * 子类对象可以访问子类的属性和方法、继承自父类的属性和方法(仍适用于属性隐藏和方法重写)
         * 子类必须是向上转型时的,否则会报错ClassCastException
         */
        子类名 子类对象名 = (子类名) 父类对象名;

        /*
         * instanceof
         * 判断对象是否属于某个类或实现某个接口,是返回true,不是返回false
         */
        boolean b = 对象名 instanceof 类名/接口名;
    }

}
package f_oop;

class Person {

    String name;

    public void show() {
        System.out.println("我是一个人");
    }

}

class Teacher extends Person {

    String school;

    public void show() {
        System.out.println("我叫" + name + ",我是一个老师");
    }

    public void teach() {
        System.out.println("我正在" + school + "进行教学……");
    }

}

class Doctor extends Person {

    String hospital;

    public void show() {
        System.out.println("我叫" + name + ",我是一个医生……");
    }

    public void operate() {
        System.out.println("我正在" + hospital + "做手术……");
    }

}

public class J_ConvertType {

    public static void main(String[] args) {
        /*
         * 向上转型
         */
        Person person = new Teacher();

        // 可以访问父类的属性和方法
        person.name = "tom";
        person.show(); // 是子类重写后的

        // 无法访问子类特有的属性和方法
        // person.school = "南京大学";
        // person.teach();

        /*
         * 向下转型
         */
        Teacher teacher = (Teacher) person;
        // Doctor doctor = (Doctor) person; // 报错:子类必须是向上转型时的

        // 可以访问子类的属性和方法、继承自父类的属性和方法
        teacher.name = "jack";
        teacher.school = "南京大学";
        teacher.show(); // 仍是子类重写后的
        teacher.teach();

        /*
         * instanceof
         */
        if (person instanceof Doctor) {
            Doctor d = (Doctor) person;
            d.name = "alice";
            d.hospital = "江苏省人民医院";
            d.operate();
        }
    }

}

二. 多态

分为:

  • 编译时多态/静态绑定:对方法进行重载,传入的实参不同,实现的功能(调用的方法)也不同。

    实现条件:方法重载。

  • 运行时多态/动态绑定:对于一个方法,父类对象作为形参,子类对象作为实参,不同子类的功能不同,该方法实现的功能也不同。

    实现条件:继承、子类重写父类的方法、向上转型。

作用:

  • 减少代码量。
  • 提高可扩展性和可维护性。
package f_oop;

class Pet {

    String name;

    String sex;

    public Pet() {
    }

    public Pet(String name, String sex) {
        this.name = name;
        this.sex = sex;
    }

    public void play() {
        System.out.println("宠物正在玩耍");
    }

}

class Cat extends Pet {

    public Cat() {
    }

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

    public void play() {
        System.out.println(name + "正在走猫步");
    }

}

class Pig extends Pet {

    public Pig() {
    }

    public Pig(String name, String sex) {
        super(name, sex);
    }

    public void play() {
        System.out.println(name + "正在跳泥坑");
    }

}

class Food {

    String name;

}

class Fish extends Food {

}

class Grass extends Food {

}

class Master {

    String name;

    public Master() {
    }

    public Master(String name) {
        this.name = name;
    }

    /*
     * 编译时多态:在编译过程就知道方法是哪个类的
     */
    // public void play(Cat cat) {
    //     System.out.println("猫走过来了");
    //
    //     cat.play();
    // }
    //
    // public void play(Pig pig) {
    //     System.out.println("猪跑过来了");
    //
    //     pig.play();
    // }

    /*
     * 运行时多态:在运行过程才知道方法是哪个类的
     */
    public void play(Pet pet) {
        pet.play();
    }

    public void feed(Pet pet, Food food) {
        System.out.println(name + "给" + pet.name + "喂" + food.name);
    }

}

public class K_Polymorphic {

    public static void main(String[] args) {
        Master master = new Master("陈孝燈");

        Cat cat = new Cat("小趴菜", "公", "游戏");
        Pig pig = new Pig("佩琪", "母");

        master.play(cat);
        master.play(pig);

        Food food = new Fish();
        food.name = "鱼";
        Grass grass = new Grass();
        grass.name = "草";

        master.feed(cat, food);
        master.feed(pig, grass);
    }

}
### Java 多态的概念 Java 中的多态是指同一个接口或类可以有多种不同的实现方式。它允许程序在运行时决定调用哪个方法,从而提高代码的灵活性和可扩展性。多态的核心机制依赖于继承、重写以及动态绑定。 #### 动态绑定 当子类覆盖父类的方法时,在运行期间会根据对象的实际类型来决定执行哪一个版本的方法[^1]。这是多态的关键特性之一。 ### 实现多态的方式 Java多态可以通过以下两种主要形式实现: 1. **方法重写(Override)** 2. **接口实现** 以下是具体示例说明如何利用 `instanceof` 运算符避免潜在异常并展示多态的应用场景。 --- ### 示例代码:Java 多态的具体应用 下面是一个完整的例子,展示了如何通过多态性和 `instanceof` 来处理不同类型的对象实例。 ```java // 定义一个基类 Animal class Animal { void makeSound() { System.out.println("Some generic sound"); } } // 子类 Dog 继承自 Animal 并重写了 makeSound 方法 class Dog extends Animal { @Override void makeSound() { System.out.println("Bark"); } // 额外的功能只属于狗 void fetchStick() { System.out.println("Fetching stick..."); } } // 子类 Cat 继承自 Animal 并重写了 makeSound 方法 class Cat extends Animal { @Override void makeSound() { System.out.println("Meow"); } // 额外的功能只属于猫 void climbTree() { System.out.println("Climbing tree..."); } } public class PolymorphismExample { public static void main(String[] args) { // 创建多个动物对象并通过向上转型存储它们 Animal myDog = new Dog(); Animal myCat = new Cat(); // 调用各自的 makeSound 方法 myDog.makeSound(); // 输出 Bark myCat.makeSound(); // 输出 Meow // 如果需要访问特定子类功能,则需使用 instanceof 和强制转换 if (myDog instanceof Dog) { ((Dog) myDog).fetchStick(); // 正确调用了 Dog 类中的特有方法 } if (myCat instanceof Cat) { ((Cat) myCat).climbTree(); // 正确调用了 Cat 类中的特有方法 } Object cValue = 42; // 假设我们有一个未知类型的变量 // 判断其实际类型并打印相应消息 System.out.println("The type of value is " + (cValue instanceof Double ? "Double" : (cValue instanceof Integer ? "Integer" : "Unknown"))); // 输出 Integer [^2] } } ``` 上述代码片段中: - 我们定义了一个通用的 `Animal` 类作为超类。 - 然后创建两个派生类 `Dog` 和 `Cat`,分别实现了自己的行为逻辑。 - 在主函数里演示了即使将这些对象赋给更广泛的父类引用 (`Animal`),仍然能够正确表现出各自的行为特征——这就是所谓的“编译看声明类型, 执行找真实类型”。 另外还加入了关于 `instanceof` 关键字使用的部分,用于确认某个对象的确切类别以便安全地进行向下造型操作而不会引发 ClassCastException 错误。 --- ###
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

睡醒的人想拯救世界

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值