【Java 基础】:三大特征之多态

✨                                                 杏花疏影里,吹笛到天明       🌏 

📃个人主页:island1314

🔥个人专栏:java学习

⛺️  欢迎关注:👍点赞 👂🏽留言 😍收藏  💞 💞 💞


🚀引言 

在前两篇博客中,我们已经讲完了面向对象程序三大特性之一的封装、继承,

【Java 基础】类和对象(构造&this&封装&static&代码块)-CSDN博客

【Java 基础】三大特征之继承-CSDN博客

下面让我们来看看多态有哪些内容吧

1.  多态概念

💢💢在Java中,多态是面向对象编程中的一个重要概念,它允许不同类型的对象对同一方法进行不同的实现。具体来说,多态性指的是通过父类的引用变量来引用子类的对象,从而实现对不同对象的统一操作

  • 多态是方法或对象具有多种形态,是面向对象的第三大特征
  • 多态前提是两个对象(类)存在继承关系,多态是建立在封装和继承基础之上的。

2. 多态实现条件

在Java中,要实现多态性,就必须满足以下条件:

  1. 继承关系
    存在继承关系的类之间才能够使用多态性。多态性通常通过一个父类用变量引用子类对象来实现。

  2. 方法重写
    子类必须重写(Override)父类的方法。通过在子类中重新定义和实现父类的方法,可以根据子类的特点行为改变这个方法的行为,如猫和狗吃东西的独特行为。

  3. 父类引用指向子类对象
    使用父类的引用变量来引用子类对象。这样可以实现对不同类型的对象的统一操作,而具体调用哪个子类的方法会在运行时多态决定

例如,下面的案例是根据猫和狗吃东西动作的不同,而实现的多态:

class Animal {public void eat() {System.out.println("动物吃东西");}
}
class Dog extends Animal {@Overridepublic void eat() {System.out.println("狗吃狗粮");}
}
class Cat extends Animal {@Overridepublic void eat() {System.out.println("猫吃猫粮");}
}public class Test {public static void main(String[] args) {Animal animal1 = new Dog(); // 父类引用指向子类对象Animal animal2 = new Cat(); // 父类引用指向子类对象animal1.eat(); // 输出:狗吃狗粮animal2.eat(); // 输出:猫吃猫粮}
}

上面代码中涉及的 Override 叫作 重写

3. 重写

3.1 重写概念

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

想要理解方法重写,需要知道以下概念:

  1. 继承关系
    重写方法是基于父类和子类之间的继承关系。子类继承了父类的方法,包括方法的名称、参数列表和返回类型。

  2. 方法签名
    重写的方法与父类的方法具有相同的方法签名,即方法的名称、参数列表和返回类型必须一致(当然,如果返回类型的对象本身的类型则可以不同,但是必须要有继承关系)。方法签名不包括方法体。

  3. @Override注解
    为了明确表明这是一个重写的方法,可以使用 @Override 注解来标记子类中的方法。该注解会在编译时检查是否满足重写条件,如果不满足会报错。

  4. 动态绑定
    通过父类引用变量调用被子类重写的方法时,会根据实际引用的对象类型,在运行时动态绑定到相应的子类方法。(下面我们会进行详细讲解)

3.2 方法重写的规则

  1. 子类在重写父类的方法时,一般必须与父类方法原型一致: 返回值类型 方法名 (参数列表) 要完全一致
  2. 被重写的方法返回值类型可以不同,但是必须是具有父子关系的
  3. 访问权限不能比父类中被重写的方法的访问权限更低。例如:如果父类方法被public修饰,则子类中重写该方法就不能声明为 protected。
  4. 父类被static、private修饰的方法、构造方法都不能被重写。
  5. 重写的方法, 可以使用 @Override 注解来显式指定. 有了这个注解能帮我们进行一些合法性校验. 例如不小心将方法名字拼写错了, 那么如果此时编译器没有在发现父类中找到该方法, 那么就会编译报错, 提示无法构成重写

注:子类中重写的方法可以调用父类中被重写的方法,使用 super 关键字。

3.3 重写和重载的区别

首先回顾重载的实现条件:

  1. 方法名称相同:重载的方法必须具有相同的名称。
  2. 参数列表不同:重载的方法的参数列表必须不同。参数列表可以通过参数的类型、个数或顺序的不同来区分重载方法
  3. 返回类型可以相同也可以不同:重载的方法可以具有相同的返回类型,也可以具有不同的返回类型。返回类型不是重载方法的区分标准。
  4. 方法所在的类中:重载方法必须定义在同一个类中
  5. 方法的访问修饰符和异常:重载方法可以具有相同的访问修饰符(如 publicprivateprotected)和抛出的异常。
区别点重写重载
定义位置定义在父类和子类之间定义在同一个类中
方法签名重写方法具有相同的名称和方法签名重载方法具有相同的名称,但方法签名(参数类型和个数)不同

继承关系

是在子类中对父类方法的重新定义和实现不涉及继承关系,可以在同一个类中定义
运行时调用是根据对象的实际类型进行动态绑定在运行时确定是根据方法的参数列表的不同进行静态绑定在编译时确定
目的用于子类重新定义父类方法的行为,以适应子类的特定需求用于在同一个类中实现相似功能但具有不同参数的方法

3.4 重写的设计原则

☘️对于已经投入使用的类,尽量不要进行修改。最好的方式是:重新定义一个新的类,来重复利用其中共性的内容,并且添加或者改动新的内容

🍀静态绑定:也称为前期绑定(早绑定),即在编译时,根据用户所传递实参类型就确定了具体调用那个方法。典型代表函数重载。

🌿动态绑定:也称为后期绑定(晚绑定),即在编译时,不能确定方法的行为,需要等到程序运行时,才能够确定具体调用那个类的方法。

  • 当调用对象方法的时候,该方法会和该对象的运行类型绑定
  • 当调用对象属性时,没有动态绑定机制,即哪里声明,哪里使用。
  • 代码示例
    //父类
    class Person {public void mission() {	System.out.println("人要好好活着!");}
    }//子类
    class Student extends Person {@Overridepublic void mission() {	System.out.println("学生要好好学习!");}
    }//演示动态绑定
    public class DynamicBinding {public static void main(String[] args) {//向上转型(自动类型转换)//程序在编译阶段只知道 p1 是 Person 类型//程序在运行的时候才知道堆中实际的对象是 Student 类型	Person p1 = new Student();  //程序在编译时 p1 被编译器看作 Person 类型//因此编译阶段只能调用 Person 类型中定义的方法//在编译阶段,p1 引用绑定的是 Person 类型中定义的 mission 方法(静态绑定)//程序在运行的时候,堆中的对象实际是一个 Student 类型,而 Student 类已经重写了 mission 方法//因此程序在运行阶段对象中绑定的方法是 Student 类中的 mission 方法(动态绑定)p1.mission();}
    }/* 结果输出*   学生要好好学习! * */

🍎🍎总的来说:重载是在同一个类中根据参数列表的不同定义多个具有相同名称但参数不同的方法,而重写是子类重新定义和实现了从父类继承的方法。重载方法通过静态绑定在编译时确定调用,重写方法通过动态绑定在运行时确定调用。重载用于实现相似功能但具有不同参数的方法,重写用于改变父类方法的行为以适应子类的需求。

4. 多态的转型

4.1 向上转型

(1)本质父类的引用指向子类的对象

(2)特点

  1. 编译类型看左边,运行类型看右边
  2. 可以调用父类的所有成员(需遵守访问权限)
  3. 不能调用子类的特有成员
  4. 运行效果看子类的具体实现

(3)语法:父类类型 对象名 = new 子类类型()

(4)样例代码:

class A1{public String name;public int age;public A1(String name, int age) {this.name = name;this.age = age;}public void eat(){System.out.println(this.name + " 父类");}public void bark1(){System.out.println(this.name + " 不构成重写父类调用");}
}class A2 extends A1 {private int num = 1;public A2(String name, int age){super(name,age);}public void eat(){ //和父类构成 重写System.out.println(this.name + " 子类");}public void bark2(){System.out.println(this.name + " 不构成重写子类调用");}
}public class Test {public static void func1(A1 a) {a.eat();a.bark1();}public static A1 func2() {A1 a = new A2("change",18);return a;}public static void main(String[] args) {// 向上转型// 1. 直接赋值A2 a2 = new A2("change", 18);A1 a1 = a2; // 子类 A2 这个引用父类这个引用所指向对象// 上下两种相同A1 a = new A2("change", 18);a.eat(); // 调用的是 子类 中的 eat() 方法//a.bark2(); // 错误:无法访问 子类中独有的方法// 2,方法的参数,传参的时候进行向上转型// 传的是子类时,只能调用子类的方法//func1(a);func1(a2);// 3. 返回值向上转型func2();}
}/* 结果输出
*   change 子类
*   change 子类
*   change 不构成重写父类调用
* */

【优缺点】

优点:让代码实现更简单灵活。
缺陷:不能调用到子类特有的方法。

4.2 向下转型

(1)本质一个已经向上转型的子类对象,将父类引用转为子类引用

(2)特点

  1. 只能强制转换父类的引用,不能强制转换父类的对象
  2. 要求父类的引用必须指向的是当前目标类型的对象
  3. 当向下转型后,可以调用子类类型中所有的成员

(3)语法:子类类型 引用名 = (子类类型) 父类引用;

(4)样例代码:

class Animal {public void eat() {System.out.println("Animal is eating.");}
}
class Dog extends Animal {@Overridepublic void eat() {System.out.println("Dog is eating");}public void bark() {System.out.println("Dog is barking");}
}class Cat extends Animal {@Overridepublic void eat() {System.out.println("Cat is eating");}public void bark() {System.out.println("Cat is barking");}
}public class Test {public static void main(String[] args) {Animal animal = new Dog();  // 向上转型animal.eat();  // 调用的是 Dog 类中的 eat() 方法// animal.bark();  // 错误:无法访问 Dog 类中独有的方法Dog dog = (Dog) animal;  // 向下转型dog.bark();  // 调用 Dog 类中的 bark() 方法}
}/* 结果输出*   Dog is eating*   Dog is barking* */

向下转型用的比较少,而且不安全,万一转换失败,运行时就会抛异常。 比如我们在上面代码进行一些修改:

 解释:这段代码在运行时出现了 ClassCastException 类型转换异常原因是 Dog 类与 Cat 类 没有继承关系,因此所创建的是 Dog 类型对象在运行时不能转换成 Cat 类型对象。

4.3 instanceof 关键字

因此Java中为了避免上述类型转换异常的问题,提高向下转型的安全性,引入了 instanceof 比较操作符,用于判断对象的类型是否为XX类型或XX类型的子类型,如果该表达式为true,则可以安全转换。 

  • 格式:对象 instanceof 类名称
  • 解释:这将会得到一个boolean值结果,也就是判断前面的对象能不能当作后面类型的实例
  • 代码示例 :
class Animal {public void eat() {System.out.println("Animal is eating.");}
}
class Dog extends Animal {@Overridepublic void eat() {System.out.println("Dog is eating");}public void bark() {System.out.println("Dog is barking");}
}class Cat extends Animal {@Overridepublic void eat() {System.out.println("Cat is eating");}public void bark() {System.out.println("Cat is barking");}
}public class Test {public static void main(String[] args) {Animal animal = new Dog();  // 向上转型if(animal instanceof Dog){ //判断对象 animal 是否是 Dog 类 的实例Dog dog = (Dog) animal;  // 向下转型dog.bark();  // 调用 Dog 类中的 bark() 方法//上面这两句也可简写为 ((Dog) animal).bark();}else if(animal instanceof  Cat){ //判断对象 animal 是否是 Cat 类 的实例((Cat)animal).bark();}}
}

5. 多态的优缺点及应用

5.1 多态的优缺点

【使用多态的好处】

  1. 灵活性和可扩展性:多态性使得代码具有更高的灵活性和可扩展性。通过使用父类类型的引用变量,可以以统一的方式处理不同类型的对象,无需针对每个具体的子类编写特定的代码。
  2. 代码复用:多态性可以促进代码的复用。可以将通用的操作定义在父类中,然后由子类继承并重写这些操作。这样一来,多个子类可以共享相同的代码逻辑,减少了重复编写代码的工作量。
  3. 可替换性:多态性允许将一个对象替换为其子类的对象,而不会影响程序的其他部分。这种可替换性使得系统更加灵活和可维护,可以方便地添加新的子类或修改现有的子类,而无需修改使用父类的代码。

  4. 代码扩展性:通过引入新的子类,可以扩展现有的代码功能,而无需修改现有的代码。这种可扩展性使得系统在需求变化时更加容易适应和扩展。

【使用多态的缺陷】

  1. 运行时性能损失:多态性需要在运行时进行方法的动态绑定,这会带来一定的性能损失。相比于直接调用具体的子类方法,多态性需要在运行时确定要调用的方法,导致额外的开销。
  2. 代码可读性下降:多态性使得代码的行为变得更加动态和不确定。在某些情况下,可能需要跟踪代码中使用的对象类型和具体的方法实现,这可能降低代码的可读性和理解性。
  3. 限制访问子类特有成员:通过父类类型的引用变量,只能访问父类及其继承的成员,无法直接访问子类特有的成员。如果需要访问子类特有的成员,就需要进行向下转型操作,这增加了代码的复杂性和维护的难度。

虽然多态性具有一些缺点,但在大多数情况下,其优点远远超过缺点,使得代码更具灵活性、可扩展性和可维护性。因此,多态性在Java编程中被广泛应用。

5.2 多态的应用

🥝多态数组

多态数组:数组的定义类型为父类类型,里面保存的实际元素类型为子类类型。
代码示例:(循环调用基类对象,访问不同派生类的方法

class Animal {public void eat() {System.out.println("Animal is eating.");}
}
class Dog extends Animal {@Overridepublic void eat() {System.out.println("Dog is eating");}
}class Cat extends Animal {@Overridepublic void eat() {System.out.println("Cat is eating");}
}
public class Test{public static void main(String[] args) {Animal[] animals = {new Animal(),new Dog(),new Cat()};for(Animal a: animals){a.eat();};}
}/* 结果输出
* Animal is eating.
* Dog is eating
* Cat is eating
* */

6. 注意事项

我们需要避免在构造方法中调用重写的方法,先来看一段代码:

class B {public B() {// do nothingfunc();}public void func() {System.out.println("B.func()");}
}class D extends B {private int num = 1;@Overridepublic void func() {System.out.println("D.func() " + num);}
}public class Test {public static void main(String[] args) {D d = new D();}
}

上面这段代码的运行结果是:D.func 0,其原因如下:

  1. 构造 D 对象的同时, 会调用 B 的构造方法。
  2. B 的构造方法中调用了 func 方法, 此时会触发动态绑定, 会调用到 D 中的 func 方法。
  3. 此时 D 对象自身还没有构造,因此 num 处在未初始化的状态,其值为 0。 如果具备多态性,num的值则应该是1。
  4. 所以在构造函数内,尽量避免使用实例方法,除了 final 和 private 方法。

结论: “用尽量简单的方式使对象进入可工作状态”, 尽量不要在构造器中调用方法(如果这个方法被子类重写, 就会触发动态绑定, 但是此时子类对象还没构造完成), 可能会出现一些隐藏的但是又极难发现的问题


💞 💞 💞那么本篇到此就结束,希望我的这篇博客可以给你提供有益的参考和启示,感谢大家支持!!!祝大家天天开心

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/415702.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

u盘格式化数据还能恢复吗?点击了解实用教程

U盘是电子数据存储设备,我们主要用它来转移数据、随身携带数据等。同时U盘在使用过程中常会遇到问题,比如U盘中毒,U盘中毒会导致里面保存的数据文件无法读取,我们需要进行U盘格式化。格式化之后的U盘才可以继续使用,那…

软件测试-Selenium+python自动化测试

目录 会用到谷歌浏览器Chrome测试,需要下载一个Chromedriver(Chrome for Testing availability)对应自己的浏览器版本号选择。 一、元素定位 对html网页中的元素进行定位,同时进行部分操作。 1.1一个简单的模板 from selenium import webdriver from selenium.webdrive…

本地部署AList并挂载小雅超集结合内网穿透实现无公网IP远程访问

文章目录 前言1. 本地部署AList2. AList挂载网盘3. 部署小雅alist3.1 Token获取3.2 部署小雅3.3 挂载小雅alist到AList中 4. Cpolar内网穿透安装5. 创建公网地址6. 配置固定公网地址 💡 推荐 前些天发现了一个巨牛的人工智能学习网站,通俗易懂&#xff…

42.接雨水

42.接雨水 给定 n 个非负整数表示每个宽度为 1 的柱子的高度图,计算按此排列的柱子,下雨之后能接多少雨水。 示例 1: 输入:height [0,1,0,2,1,0,1,3,2,1,2,1] 输出:6 解释:上面是由数组 [0,1,0,2,1,0,1…

一文搞懂微服务架构之限流

前置知识 限流是通过限制住流量大小来保护系统,能够解决异常突发流量打崩系统的问题。例如常见的某个攻击者在攻击你维护的系统,那么限流就是极大程度上保护住你的系统。 算法 限流算法也可以像负载均衡算法那样,划分成静态算法和动态算法…

【Java】—— Java面向对象高级:关键字static的使用

目录 1. 关键字:static 1.1 类属性、类方法的设计思想 1.2 static关键字 1.3 静态变量 1.3.1 语法格式 1.3.2 静态变量的特点 1.4 静态方法 1.4.1 语法格式 1.4.2 静态方法的特点 1.5 练习 1. 关键字:static 回顾类中的实例变量(即…

Android - Windows平台下Android Studio使用系统的代理

这应该是第一篇Android的博文吧。以后应该会陆续更新的。记录学习Android的点点滴滴。 之前也看过,不过看完书就忘了,现在重拾Android,记录学习历程。 为何要用代理 因为更新gradle太慢了。 如何使用系统的代理 先找到系统代理的ip和端口。…

C++学习笔记----6、内存管理(一)---- 使用动态内存(3)

3.2、对象数组 对象数组与原型/基础类型的数组没有什么不同,除了元素的初始化之外。当你使用new[N]去分配N个对象,就把N个连续的块空间分配出去了,每一个块空间可以放一个单独的对象。对于对象数组,New[]对每一个对象自动调用0参数…

激光雷达产品介绍

与传统激光雷达线性重复式的扫描方式不同,Livox mid系列激光雷达扫描路径不会重复。且视场中激光照射到的区域面积会随时间增大,这就意味着视场覆盖率随时间推移而显著提高。 内容参考自《解构大疆旗下 Livox Mid 激光雷达非重复扫描技术》作者&#xff…

【C++11(一)之入门基础)】

文章目录 C简介统一的列表初始化{}初始化 std::initializer_liststd::initializer_list是什么类型:std::initializer_list使用场景: 声明autodecltypenullptr STL中一些变化 C简介 在2003年C标准委员会曾经提交了一份技术勘误表(…

#ARM开发 笔记

课程介绍 ARM开发 --> Linux移植 --> 驱动开发 前后联系:ARM和系统移植为驱动开发学习做准备工作 所需知识:C语言基础及STM32需要的硬件知识 学习方法 学习流程、思想和解决问题的方法即可 知道驱动的基本框架以及基本开发要求 底层课程导学 接口技…

linux小程序-进度条

文章目录 pro.hpro.cmain.cmakefile测试 pro.h #pragma once#include <stdio.h>typedef void(*callback_t)(double, double);void probar(double total, double current);pro.c #include "pro.h" #include <string.h> #include <unistd.h> #defi…

webshell绕过样本初体验

目录 一&#xff1a;前景 二&#xff1a;样本 样本一&#xff1a; 样本二&#xff1a; 样本三&#xff1a; 样本4&#xff1a; 样本5&#xff1a; 一&#xff1a;前景 在我们日常的网站中百分之一百是存在一些安全设备来拦截我们的webshell的&#xff0c;一般情况…

Kafka大厂面试14问(附答案)

怎么保证顺序消费&#xff1f; 同一个生产者发送到同一分区的消息&#xff0c;先发送的比后发送的offset要小。同一生产者发送到不同分区的消息&#xff0c;消息顺序无法保证。 怎么解决这个问题&#xff1f; 给一个topic只设置一个分区 相同key会发给一个分区 怎么保证幂…

[有彩蛋]大模型独角兽阶跃星辰文生图模型Step-1X上线,效果具说很炸裂?快来看一手实测!

先简单介绍一下阶跃星辰吧 公司的创始人兼CEO是姜大昕博士&#xff0c;他在微软担任过全球副总裁&#xff0c;同时也是微软亚洲互联网工程研究院的副院长和首席科学家。 2024年3月&#xff0c;阶跃星辰发布了Step-2万亿参数MoE语言大模型预览版&#xff0c;这是国内初创公司首…

Centos7通过reposync搭建本地Yum源

目录 1. 服务端搭建 1.1. 安装相关软件包 1.2. 加载几个常用的yum源 1.3. 创建文件保存目录 1.4. 把各仓库同步到本地 1.5. 生成仓库信息 1.6. 定时任务更新仓库 1.7. nginx配置下载服务 1.8. 内网测试nginx服务配置是否正确 2. 客户端配置 前言&#xff1a;之前使用…

Nginx负载均衡数据流分析

1、各机器ip信息 客户端IP&#xff1a;192.168.3.239 Nginx代理服务器IP&#xff1a;192.168.3.241 服务端IP&#xff1a;192.168.3.238 2、架构图&#xff08;略&#xff09; 3、 下图是在服务端上面的抓包分析。 下图是在客户端上面的抓包分析&#xff1a; 下图是在代理服务…

动态路由和路由导航守卫及其案例分析

为什么需要动态路由&#xff1f; 动态路由其实用的不多&#xff0c;在实际开发中&#xff0c;如果遇到权限分配问题&#xff0c;比如对于一个公司人员的后台管理系统&#xff0c;那对不同成员的权限肯定不同&#xff0c;对于人事部&#xff0c;他们有权限进入成员表对人员的流…

PHP8、ThinkPHP8框架中间的应用教程详解

前言 虽然PHP的落幕的话题一直不绝&#xff0c;但是实际在WEB端项目中PHP占有率达到了70%以上&#xff0c;一直在WEB一枝独秀&#xff0c;它以快速、高效的开发闻名&#xff0c;出圈了几十年&#xff0c;等待只是下一次的涅槃。而经过PHP8、PHP9的演变发展&#xff0c;PHP逐渐…

【Linux网络编程】协议|OSI模型|TCP/IP模型|局域网通信|跨网络通信|地址管理|流程图

目录 ​编辑 一&#xff0c;协议 协议分层 二&#xff0c;OSI七层模型 三&#xff0c;TCP/IP五层&#xff08;或四层&#xff09;模型 TCP/IP各个层次一些名词解释 为什么要有TCP/IP协议 TCP/IP协议栈与操作系统的宏观关系示意图 四&#xff0c;网络传输基本流程 局…