什么是多态?

简介: 什么是多态?

@TOC

一、多态是什么?

多态是同一个行为具有多个不同表现形式或形态的能力。
多态就是同一个接口,使用不同的实例而执行不同操作,如图所示:
在这里插入图片描述

二、多态的必要条件

2.1 初始多态

要实现多态,以下三大条件缺一不可:

  1. 必须在继承体系下
  2. 子类必须要对父类中方法进行重写
  3. 通过父类的引用调用重写的方法

在这里插入图片描述

class Shape {
    void draw() {

    }
}
class Square extends Shape {
    @Override
    void draw() {
        System.out.println("画Square!");
    }
}
class Circle extends Shape {
    @Override
    void draw() {
        System.out.println("画Circle!");
    }
}
class Triangle extends Shape {
    @Override
    void draw() {
        System.out.println("画Triangle!");
    }
}
public class Test {
    public static void draw(Shape shape) {
        shape.draw();
    }
    public static void main(String[] args) {
        Circle circle = new Circle();
        Square square = new Square();
        draw(circle);
        draw(square);
    }
}

在这里插入图片描述
在这里实现了同一个方法同一引用,因为传入对象的不同,产生了不同的效果,这种不同的体现就是多态的体现.
在这里插入图片描述

2.2 多态的优缺点

优点:

  1. 消除类型之间的耦合关系
  2. 可替换性
  3. 可扩充性
  4. 接口性
  5. 灵活性
  6. 简化性

缺点:
1.属性没有多态性
当父类和子类属性同名时,只能访问父类的
2.构造方法没有多态性.

2.3 重写

重写(override):也称为覆盖。重写是子类对父类非静态、非private修饰,非final修饰,非构造方法等的实现过程进行重新编写, 返回值和形参都不能改变。即外壳不变,核心重写!重写的好处在于子类可以根据需要,定义特定于自己的行为。 也就是说子类能够根据需要实现父类的方法。

在这里插入图片描述
在重写的时候可以加上@Override关键字,系统会默认帮我们检查重写的方法是否正确,如果父类没有此类方法,就会编译报错.
|区别|重写|重载|

方法名 不能修改 不能修改
参数列表 一定不能修改 必须修改
返回类型 不能修改或者构成父子关系 可以修改
访问限定符 不能增强限制 可以修改

在这里插入图片描述
对于已经投入使用的类,尽量不要进行修改。最好的方式是:重新定义一个新的类,来重复利用其中共性的内容,
并且添加或者改动新的内容。
在这里插入图片描述
静态绑定:也称为前期绑定(早绑定),即在编译时,根据用户所传递实参类型就确定了具体调用那个方法。典型代表==函数重载。==
动态绑定:也称为后期绑定(晚绑定),即在编译时,不能确定方法的行为,需要等到程序运行时,才能够确定具体调用那个类的方法。

三、转型

3.1 向上转型

Shape shape = new Circle();

向上转型就是new 一个子类对象,让父类对象来使用
在这里插入图片描述

class Animal {
    String neme;
    int age;

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

    public void eat(){

   }
}

class Cat extends Animal {
    public Cat(String neme, int age) {
        super(neme, age);
    }

    public void eat() {
        System.out.println("吃鱼");
    }
    public void work() {
        System.out.println("抓老鼠");
    }
}

class Dog extends Animal {
    public Dog(String neme, int age) {
        super(neme, age);
    }

    public void eat() {
        System.out.println("吃骨头");
    }
    public void work() {
        System.out.println("看家");
    }
}

1.直接赋值:

public static void main(String[] args) {
        Animal cat = new Cat("咪咪",18);//直接将子类对象赋值给父类对象
    }

2.方法传参:

public static void eat(Animal animal) {
        //方法传参:形参为父类引用,可以接受子类对象
        animal.eat();
    }

3.方法返回:

public static Animal reAnimal(String type) {
        //返回一个子类对象
        if("狗".equals(type)) {
            return new Dog("旺财",18);
        }else if("猫".equals(type)) {
            return new Cat("咪咪",18);
        }else {
            return null;
        }
    }

向上转型可以让代码变得更加简单灵活,但无法访问子类特有的属性方法.

3.2 向下转型

在进行向上转型后,无法调用子类的方法,但有时可能需要调用子类方法,此时在还原为子类对象,成为向下转型.

==放下转型具有不安全性==

在这里插入图片描述
向下转型用的比较少,而且不安全,万一转换失败,运行时就会抛异常。Java中为了提高向下转型的安全性,引入了 instanceof ,如果该表达式为true,则可以安全转换。

public static void main(String[] args) {
        Animal animal = new Dog("旺财",18);
        if(animal instanceof Cat) {
            Cat cat = (Cat)animal;
        }
        else if(animal instanceof Dog) {
            Dog dog = (Dog)animal;
        }
    }

通过向下转型后就可以正常的调用子类方法.

3.3 避免在构造方法中调用重写方法.

class A {
    public A() {
        fun();
    }
    public void fun() {
        System.out.println("A:fun");
    }
}
class B extends A {
    int count = 10;

    @Override
    public void fun() {
        System.out.println("B:fun"+count);
    }

    public static void main(String[] args) {
        B b = new B();
    }
}

这样的代码会出现很多问题.它先执行父类的构造方法,然后在构造方法中执行子类的重写方法,但因为子类还没有执行构造方法,子类的属性还没有被加载,所以这里输出的count不是10,而是默认值0.
在这里插入图片描述
在这里插入图片描述

结论: "用尽量简单的方式使对象进入可工作状态", 尽量不要在构造器中调用方法(如果这个方法被子类重写, 就会触发动态绑定, 但是此时子类对象还没构造完成), 可能会出现一些隐藏的但是又极难发现的问题.
目录
相关文章
|
SQL 弹性计算 安全
ECS权益问题之学生权益无法续费如何解决
ECS(Elastic Compute Service,弹性计算服务)是云计算服务提供商提供的一种基础云服务,允许用户在云端获取和配置虚拟服务器。以下是ECS服务使用中的一些常见问题及其解答的合集:
|
4月前
|
安全 Java 开发者
Java集合框架:详解Deque接口的栈操作方法全集
理解和掌握这些方法对于实现像浏览器后退功能这样的栈操作来说至关重要,它们能够帮助开发者编写既高效又稳定的应用程序。此外,在多线程环境中想保证线程安全,可以考虑使用ConcurrentLinkedDeque,它是Deque的线程安全版本,尽管它并未直接实现栈操作的方法,但是Deque的接口方法可以相对应地使用。
272 12
|
API 数据安全/隐私保护 iOS开发
利用uni-app 开发的iOS app 发布到App Store全流程
利用uni-app 开发的iOS app 发布到App Store全流程
234 1
|
12月前
|
存储 自然语言处理 机器人
基于的Qwen模型的智能客服Discord机器人,使用🐫 CAMEL、SambaNova、Firecrawl和Qdrant实现RAG Agent
基于Qwen模型的智能客服Discord机器人,使用CAMEL、SambaNova、Firecrawl和Qdrant实现RAG Agent。构建了一个能够处理复杂问题并能进行快速响应的强大聊天机器人。该机器人可在Discord平台上运行,支持实时对话和语义搜索,提供准确、全面的回答。项目包含详细的安装步骤、代码示例及集成指南,适合开发者快速上手。
|
搜索推荐 Linux
深入理解Linux操作系统的启动过程
本文旨在揭示Linux操作系统从开机到完全启动的神秘面纱,通过逐步解析BIOS、引导加载程序、内核初始化等关键步骤,帮助读者建立对Linux启动流程的清晰认识。我们将探讨如何自定义和优化这一过程,以实现更高效、更稳定的系统运行。
|
9月前
|
人工智能 前端开发 Java
DDD四层架构和MVC三层架构的个人理解和学习笔记
领域驱动设计(DDD)是一种以业务为核心的设计方法,与传统MVC架构不同,DDD将业务逻辑拆分为应用层和领域层,更关注业务领域而非数据库设计。其四层架构包括:Interface(接口层)、Application(应用层)、Domain(领域层)和Infrastructure(基础层)。各层职责分明,避免跨层调用,确保业务逻辑清晰。代码实现中,通过DTO、Entity、DO等对象的转换,结合ProtoBuf协议,完成请求与响应的处理流程。为提高复用性,实际项目中可增加Common层存放公共依赖。DDD强调从业务出发设计软件,适应复杂业务场景,是微服务架构的重要设计思想。
|
9月前
|
存储 编译器 C++
【c++】多态(多态的概念及实现、虚函数重写、纯虚函数和抽象类、虚函数表、多态的实现过程)
本文介绍了面向对象编程中的多态特性,涵盖其概念、实现条件及原理。多态指“一个接口,多种实现”,通过基类指针或引用来调用不同派生类的重写虚函数,实现运行时多态。文中详细解释了虚函数、虚函数表(vtable)、纯虚函数与抽象类的概念,并通过代码示例展示了多态的具体应用。此外,还讨论了动态绑定和静态绑定的区别,帮助读者深入理解多态机制。最后总结了多态在编程中的重要性和应用场景。 文章结构清晰,从基础到深入,适合初学者和有一定基础的开发者学习。如果你觉得内容有帮助,请点赞支持。 ❤❤❤
1184 0
|
调度 Android开发 UED
Android经典实战之Android 14前台服务适配
本文介绍了在Android 14中适配前台服务的关键步骤与最佳实践,包括指定服务类型、请求权限、优化用户体验及使用WorkManager等。通过遵循这些指南,确保应用在新系统上顺畅运行并提升用户体验。
922 6