类和对象(4)——多态:方法重写与动态绑定、向上转型和向下转型、多态的实现条件

目录

1. 向上转型和向下转型

1.1 向上转型

1.2 向下转型

1.3 instanceof关键字

2. 重写(overidde)

2.1 方法重写的规则

2.1.1 基础规则

2.1.2 深层规则

2.2 三种不能重写的方法

final修饰

private修饰 

static修饰

3. 动态绑定

3.1 动态绑定的概念

3.2 动态绑定与静态绑定

4. 多态

4.1 多态的实现场景

1. 基类形参方法 

2. 基数组

4.2 多态缺陷

1. 属性(字段)没有多态性 

2. 向上转型不能使用子类特有的方法

3. 构造方法没有多态性


上一篇文章中,我们深度学习了继承的概念与实现。在继承篇中,我们最重要的就是“弄清楚通过子类实例变量来访问与父类相同的成员会怎么样”;而在多态篇中,最核心的内容就是“弄清楚通过父类实例变量来访问与子类相同的方法会怎么样”。

1. 向上转型和向下转型

在了解多态之前,我们还要补充几个知识点,这首先就是向上转型和向下转型。

1.1 向上转型

向上转型是指将一个子类对象的引用赋值给一个父类类型的实例变量

语法格式:

父类类型 对象名 = new 子类类型();

//例如 Animal animal = new Cat("小咪", 2);

animal是父类类型的实例变量,但引用的是一个子类Cat对象,因为这是从小范围向大范围的转换。类似基础数据类型中的隐式类型转换(例如长整形long接收整形int的数据)


向上转型的3种使用场景:

  1. 直接赋值:子类对象的引用直接赋值给父类类型的实例变量
  2. 方法传参:子类对象的引用作为参数,传递给方法中的父类类型的形参
  3. 方法返回:方法的返回类型是父类类型返回的值是子类类型

例如:

public class TestAnimal {// 2. 方法传参:形参为父类型引用,可以接收任意子类的对象public static void eatFood(Animal a){a.eat();}// 3. 作返回值:返回任意子类对象public static Animal buyAnimal(String var){if("狗".equals(var) ){return new Dog("狗狗",1);}else if("猫" .equals(var)){return new Cat("猫猫", 1);}else{return null;}}public static void main(String[] args) {Animal cat = new Cat("元宝",2);   // 1. 直接赋值:子类对象赋值给父类对象Dog dog = new Dog("小七", 1);eatFood(cat);eatFood(dog);Animal animal = buyAnimal("狗");animal.eat();animal = buyAnimal("猫");animal.eat();}}

1.2 向下转型

向下转型是将父类对象强制转换为子类对象的过程,需要用到类型转换运算符( ) 。

【注意】

  • 只能对已向上转型的对象进行向下转型:不能直接将一个父类对象强制转换为子类对象,除非这个父类对象实际上是子类对象的向上转型。也就是说,必须先创建一个子类对象,然后将其向上转型为父类对象,最后再进行向下转换。
  • 向上转型的子类类型 与 向下接收的子类类型必须一致

例如:

先看父类和子类的具体代码:

public class Animal {public String name;public Animal(String name){this.name = name;}public void eat(){System.out.println(name+"在吃东西");}
}public class Dog extends Animal{public Dog(String name){super(name);}public void eat(){System.out.println(name+"在吃狗粮");}//Dog类的专属方法public void bark(){System.out.println(name+"在汪汪叫");}
}public class Cat extends Animal{public Cat(String name){super(name);}public void eat(){System.out.println(name+"在吃猫粮");}//Cat类的专属方法public void mew(){System.out.println(name+"在喵喵叫");}
}

测试1:父类实例animal是Dag类的向上转型,再让animal向下转型传给子类实例dog。

public class Test {public static void main(String[] args) {Animal animal = new Dog("旺财");Dog dog;dog = (Dog) animal;dog.bark();}
}

运行成功


测试2:父类实例animal是Dag类的向上转型,再让animal向下转型传给子类实例cat。(向上转型的子类与向下接收的子类不一致

public class Test {public static void main(String[] args) {Animal animal = new Dog("旺财");Cat cat;cat = (Cat) animal;//抛出异常}
}

抛出异常


1.3 instanceof关键字

向下转型用的比较少,而且不安全,万一转换失败,运行时就会抛异常。Java中为了提高向下转型的安全性,引入了instanceof关键字。

语法:

        Object  instanceof  ClassName

其中,object 是要测试的对象或实例变量,ClassName 是要测试的类名。

作用和返回值:

如果 object 是 ClassName 的实例其子类的实例,则表达式返回 true;否则返回 false

有了instanceof,我们向下转型就可以更安全了:

public class TestAnimal {public static void main(String[] args) {Cat cat = new Cat("元宝",2);Dog dog = new Dog("小七", 1);// 向上转型Animal animal = cat;animal.eat();animal = dog;animal.eat();if(animal instanceof Cat){    //检查类型cat = (Cat)animal;cat.mew();}if(animal instanceof Dog){    //检查类型dog = (Dog)animal;dog.bark();}}}

animal最后是Dog类的引用,所以通过了第2个检查,由dog接收animal的向下转型。

2. 重写(overidde)

2.1 方法重写的规则

2.1.1 基础规则

方法重写:也称为方法覆盖,即外壳不变,核心重写。

  1. 子类在重写父类的方法时,一般必须与父类的方法原型一致【返回值类型、方法名、参数列表完全一致】。 
  2. 重写的方法,可以在子类方法头的上一行使用“ @Override ”注解来显式指定。有了这个注解能帮我们进行一些合法性校验(例如不小心将方法名字拼写错了,比如eat写成 aet,那么此时编译器就会发现父类中没有 aet 方法, 就会编译报错, 提示无法构成重写)

例如:

如果显示指定@overidde,但方法与父类原型不一致的话,系统会报错:

2.1.2 深层规则

  1. 返回值类型:其实子类重写的方法返回类型也可以与父类不一样,但是必须是具有父子关系的。       
    1.  该要求其实隐藏了一个情况,那就是此时的方法返回值类型是类类型的。
    2.  对于这种情况的方法重写,父类方法的返回值类型 必须是 子类方法的返回值类型基类
  2. 访问限定符:访问权限不能比父类中被重写的方法的访问权限更低。(即子类重写的方法访问权限可以更宽松,不能更严格)

对于“返回值类型”要求的举例:

​class Parent {public Number display() {return 42; // 返回一个Integer类型的值}
}class Child extends Parent {// 重写父类的display方法,并改变返回类型为Double,这是允许的,因为Double是Number的子类型@Overridepublic Double display() {return 42.0;}
}public class Test {public static void main(String[] args) {Parent parent = new Parent();System.out.println(parent.display()); // 输出: 42Child child = new Child();System.out.println(child.display()); // 输出: 42.0}
}

Parent类的display方法的返回类型是Number类,Child类的display方法的返回类型是Double类。其中Parent类是Child类的父类,Number类又是Double类和Interger类的父类,这符合方法重写的深层规则。

如果该例子中的 方法返回值类型的父子关系反过来 会报错:


对于“访问限定符”要求的举例:


关于方法重写还有更深层更严格的规定,这些规定与异常、线程等有关。本章重点是继承与多态,所以不再具体展开。

2.2 三种不能重写的方法

如果父类方法被final、private或static修饰,则子类不能重写该方法。

final修饰

final:

final修饰成员方法时,就是用来防止该方法被子类重写 或者 不想让该方法被重写。

例如:


private修饰 

private:

父类的方法被private修饰时,说明这个方法是父类私有的,子类也没有办法去访问该方法。

  • 如果private修饰了父类的方法,子类又写了一个与父类方法原型一样的方法,系统并不会报错
    (因为系统检查方法重写时,会自动把父类的私有方法忽略掉)
  • 如果private修饰了子类的方法,父类又有一个与子类方法原型一样的非private方法,那么系统会报错
    (此时系统会认为你想要让子类重写父类方法,又因为重写后的方法是私有的而父类的方法非私有,所以会提醒你“分配了更低的访问权限”并报错)

例1:

父类方法是私有的,build没有问题

例2:

子类方法是私有的,父类方法原型与子类一致。系统认为你要重写,但子类的方法权限更低,所以报错:


static修饰

static:

静态方法是在类加载时就绑定到类本身,而不是在运行时绑定到具体的对象实例,所以static修饰的方法不能被重写。即静态方法不能实现动态绑定,也就不能被覆盖(重写)。

例如:

虽然静态方法是可以被继承的,但如果子类定义了一个与父类相同签名的静态方法这只是对父类静态方法的一种隐藏,而非真正意义上的重写。

  • 子类对象向上转型后,当通过父类实例变量引用调用该方法时,仍然会执行父类的静态方法,而不是子类的静态方法。

例如:

class Parent {static void display() {System.out.println("Parent display method");}
}class Child extends Parent {static void display() {System.out.println("Child display method");}
}public class Test {public static void main(String[] args) {Parent parent1 = new Parent();parent1.display();  //调用父类静态方法Child child = new Child();child.display();   //调用子类静态方法Parent parent2 = new Child();   //向上转型parent2.display();  //调用父类静态方法}
}

3. 动态绑定

3.1 动态绑定的概念

刚刚在解释static修饰方法时,我们提到了一个词叫动态绑定。下面让我们看看什么是动态绑定。

概念:

动态绑定也叫后期绑定,是指在运行时根据对象的实际类型来确定调用哪个方法,而不是在编译时就决定。当一个父类引用指向其子类的对象,并且通过该引用调用一个被重写的方法时,会在运行时根据对象的实际类型来调用相应的方法实现,这就是重写方法的动态绑定。

动态绑定重写方法的实现条件:

  1. 存在继承关系必须有一个基类(父类)和至少一个派生类(子类),子类继承自父类。
  2. 方法重写子类要实现父类中至少一个方法的重写。
  3. 向上转型在程序中存在向上转型的情况,即把子类对象的引用赋值给父类的实例变量

例如:

class Animal {public String name;public Animal(String name){this.name = name;}public void eat(){System.out.println(name+"在吃东西");}
}public class Dog extends Animal{public Dog(String name){super(name);}public void eat(){System.out.println(name+"在吃狗粮");}
}public class Cat extends Animal{public Cat(String name){super(name);}public void eat(){System.out.println(name+"在吃猫粮");}
}public class Test {public static void main(String[] args) {Animal animal = new Animal("动物");//animal动态绑定到Animal类animal.eat();animal = new Dog("小狗");//animal动态绑定到Dog类animal.eat();animal = new Cat("小猫");//animal动态绑定到Cat类animal.eat();}
}

3.2 动态绑定与静态绑定

静态绑定也称为早期绑定:是指在程序编译时就已经确定了方法调用的具体对象和方法实现。与动态绑定相对应,静态绑定不需要运行时进行额外的判断和查找来确定调用哪个方法。

静态绑定的适用情况

  • 基本数据类型的方法调用(可重载的方法):对于基本数据类型的操作方法,如数学运算等,通常是静态绑定。例如,int a = 5; int b = 10; int c = a + b; 中 + 运算符对应的加法方法是在编译时就确定的。
  • 私有方法、静态方法和 final 方法:这些方法不能被重写或具有特殊的性质,所以它们的调用可以在编译时确定。例如,class Example { private void privateMethod() {...} static void staticMethod() {...} final void finalMethod() {...} } 中的私有方法、静态方法和 final 方法都是静态绑定的。
  • 构造方法:构造方法在创建对象时被调用,每个类都有特定的构造方法,且在编译时就可以确定是哪个类的构造方法会被调用。例如,new Example() 会调用 Example 类的构造方法,这是在编译时就已经决定的。

方法重载(静态绑定)是一个类的多态性表现【例如工具类Arrays】,而方法重写(动态绑定)是子类与父类间的多态性的表现。

4. 多态

多态的概念:

去完成某个行为时,当不同的对象去完成时会产生出不同的状态。又或者同一件事情,发生在不同对象身上,就会产生不同的结果

打个比方,语文老师要求同学们背一首诗,同学A背了一首李白的诗、同学B背了一首杜甫的诗、同学C背了一首李清照的诗……每个同学背的诗都不同,但不管怎么说他们都完成了“背一首诗”的任务,这就是多态。

4.1 多态的实现场景

在java中要实现多态,必须要满足如下几个条件,缺一不可:

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

下面我来介绍两种常见的多态实现。

1. 基类形参方法 

基类形参方法:指的是形参数据类型为基类类型的方法。

该方法的形参的类型是父类类型,我们一般在该方法中使用被重写的方法。不同的子类实例变量传参进去并发生向上转型,该基类形参方法就能够通过动态绑定来调用不同的重写方法,从而实现多态。

例如:

//有继承关系的类
public class Animal {public String name;public Animal(String name){this.name = name;}public void eat(){System.out.println(name+"在吃东西");}
}public class Dog extends Animal{public Dog(String name){super(name);}public void eat(){System.out.println(name+"在吃狗粮");}
}public class Cat extends Animal{public Cat(String name){super(name);}public void eat(){System.out.println(name+"在吃猫粮");}
}
————————————————————————————————————————————————————————
————————————————————————————————————————————————————————
//含基类形参方法的类
public class Test {public void eat(Animal animal){ //基类形参方法animal.eat();}public static void main(String[] args) {Test test = new Test();    //如果Test的eat方法是静态方法,那么可以不用new一个Test对象test.eat(new Animal("小动物"));test.eat(new Dog("小狗"));test.eat(new Cat("小猫"));}
}


这种方法有点类似C语言中的函数指针和回调函数的用法。详细请看《指针之旅(4)—— 指针与函数:函数指针、转移表、回调函数》

2. 基数组

基数组:指的是数组元素的类型都是基类类型。

由于可以向上转型,在基数组中可以存放子类对象,从而实现多态。

(有点类似C语言中的函数指针数组)

例如:

​//Animal类、Dog类和Cat类的内容如上面的一致
public class Test {public static void main(String[] args) {//基数组animalsAnimal[] animals = {new Animal("小动物"), new Dog("小狗"), new Cat("小猫")};for(Animal x: animals){x.eat();    //临时变量x通过动态绑定实现多态}}
}

如果有新的动物增加,我们可以在基数组animals中添加,这就是多态的好处,十分便捷。

如果不基于多态来实现刚刚的代码内容,我们需要多个if-else语句,如下:

在这种情况下,如果要增加一个动物,不仅字符串数组animals要变,而且在for-each循环中还要加多一条else-if语句,十分不便。

4.2 多态缺陷

1. 属性(字段)没有多态性 

当父类和子类都有同名属性的时候,通过父类实例变量引用只能引用父类自己的成员属性

例如:

public class Parent {public String str = "parent";
}public class Child extends Parent{public String str = "child";
}public class Test {public static void main(String[] args) {Parent parent = new Parent();System.out.println(parent.str);//打印parentChild child = new Child();System.out.println(child.str);//打印childparent = child;     //向上转型System.out.println(parent.str);//属性没有多态性,打印的还是父类的str}
}

2. 向上转型不能使用子类特有的方法

方法调用在编译时进行类型检查,编译器只检查引用变量类型中定义的方法,而不考虑实际对象的类型

例如,这里子类Dog比父类Animal类多了一个特殊方法bark()。如果用Animal类型的实例变量来接收Dog类的对象,我们会发现无法通过该实例变量调用bark方法:

3. 构造方法没有多态性

父类的构造方法中调用一个被重写的方法时,实际执行的是子类中的实现。然而,此时子类可能还未完成初始化,其成员变量尚未赋值或处于默认状态,这就可能导致程序行为不确定,甚至引发错误。

父类构造方法中如果调用了被重写的方法,那么该重写的方法使用的是子类的方法

我们创建两个类, B 是父类, D 是子类. D 中重写 func 方法. 并且在 B 的构造方法中调用 func:

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

  • 构造 D 对象的同时,会调用 B 的构造方法.
  • B 的构造方法中调用了 func 方法, 此时会触发动态绑定,会调用到 D 中的 func 。此时 D 对象自身还没有构造,此时 num 处在未初始化的状态,值为 0.
  • 如果具备多态性,num的值应该是10.

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


本期分享完毕,感谢大家的支持Thanks♪(・ω・)ノ

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

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

相关文章

AUTOSAR从入门到精通-车身控制系统BCM(三)

目录 前言 算法原理 什么是车身控制模块BCM 1. BCM ECU的工作原理 a. 硬件架构 b. 控制逻辑 BCM带来的好处 车身控制模块(BCM)的功用 车身控制模块(BCM)能够控制的车身功能系统 BCM的各项功能 1.1内外部灯光控制 1.2 雨刮系统 1.3 车身防盗报警系统 1.4 车锁…

16届蓝桥杯寒假刷题营】第2期DAY5IOI赛

3.小蓝小彬的代码挑战 - 蓝桥云课 问题描述 在蓝桥杯大赛中,小蓝和小彤是一对好朋友。他们在比赛中遇到了一个有趣的挑战。这个挑战是给定一个由大写字母组成的代码,他们需要找出这串代码中有多少个子序列LQB。小蓝和小彬都很聪明,他们想到…

51单片机入门_02_C语言基础0102

C语言基础部分可以参考我之前写的专栏C语言基础入门48篇 以及《从入门到就业C全栈班》中的C语言部分,本篇将会结合51单片机讲差异部分。 课程主要按照以下目录进行介绍。 文章目录 1. 进制转换2. C语言简介3. C语言中基本数据类型4. 标识符与关键字5. 变量与常量6.…

windows系统如何检查是否开启了mongodb服务

windows系统如何检查是否开启了mongodb服务!我们有很多软件开发,网站开发时候需要使用到这个mongodb数据库,下面我们看看,如何在windows系统内排查,是否已经启动了本地服务。 在 Windows 系统上,您可以通过…

【Java】微服务找不到问题记录can not find user-service

一、问题描述 运行网关微服务与用户微服务后,nacos服务成功注册 但是测试接口的时候网关没有找到相关服务 二、解决方案 我先检查了pom文件确定没问题后查看配置文件 最后发现是配置里spring.application.namexxx-user里面服务的名字后面多了一个空格 三、总结…

Java设计模式:行为型模式→策略模式

Java 策略模式详解 1. 定义 策略模式(Strategy Pattern)是一种行为型设计模式,它定义了一系列的算法,将每一个算法封装起来,并使它们可以互相替换。策略模式让算法的变化独立于使用算法的客户。通过这种模式&#xf…

c++学习第十四天

提示:以下是本篇文章正文内容,下面案例可供参考。 //力扣代码 class Solution {const char* numStrArr[10]{"","","abc","def","ghi","jkl","mno","pqrs","tuv&q…

Python Matplotlib库:从入门到精通

Python Matplotlib库:从入门到精通 在数据分析和科学计算领域,可视化是一项至关重要的技能。Matplotlib作为Python中最流行的绘图库之一,为我们提供了强大的绘图功能。本文将带你从Matplotlib的基础开始,逐步掌握其高级用法&…

Vue平台开发三——项目管理页面

前言 对于多个项目的使用,可能需要进行项目切换管理,所以这里创建一个项目管理页面,登录成功后跳转这个页面,进行选择项目,再进入Home页面展示对应项目的内容。 一、实现效果图预览 二、页面内容 功能1、项目列表展…

常见字符串相关题目

找往期文章包括但不限于本期文章中不懂的知识点: 个人主页:我要学编程(ಥ_ಥ)-CSDN博客 所属专栏: 优选算法专题 目录 14.最长公共前缀 5.最长回文子串 67.二进制求和 43.字符串相乘 14.最长公共前缀 题目: 编写一个函数来查…

Seed Edge- AGI(人工智能通用智能)长期研究计划

Seed Edge 是字节跳动豆包大模型团队推出的 AGI(人工智能通用智能)长期研究计划12。以下是对它的具体介绍1: 名称含义 “Seed” 即豆包大模型团队名称,“Edge” 代表最前沿的 AGI 探索,整体意味着该项目将在 AGI 领域…

Java定时任务实现方案(四)——Spring Task

Spring Task 这篇笔记,我们要来介绍实现Java定时任务的第四个方案,使用Spring Task,以及该方案的优点和缺点。 ​ Spring Task是Spring框架提供的一个轻量级任务调度框架,用于简化任务调度的开放,通过注解或XML配置的…

(15)基于状态方程的单相自耦变压器建模仿真

1. 引言 2. 单相降压自耦变压器的状态方程 3. 单相降压自耦变压器的simulink仿真模型 4. 实例仿真 5. 总结 1. 引言 自耦变压器的原边和副边之间存在直接的电气连接,所以功率是通过感应和传导从原边转移到副边的,这与双绕组变压器不同,后者的原边和副边是电气隔离的。从…

Highcharts 柱形图:深入解析与最佳实践

Highcharts 柱形图:深入解析与最佳实践 引言 Highcharts 是一个功能强大的图表库,它允许用户轻松地在网页上创建各种类型的图表。其中,柱形图因其直观的展示方式,在数据分析、业务报告等领域得到了广泛应用。本文将深入解析 Highcharts 柱形图,包括其基本用法、高级特性…

【C++基础】多线程并发场景下的同步方法

如果在多线程程序中对全局变量的访问没有进行适当的同步控制(例如使用互斥锁、原子变量等),会导致多个线程同时访问和修改全局变量时发生竞态条件(race condition)。这种竞态条件可能会导致一系列不确定和严重的后果。…

网络安全 | F5-Attack Signatures-Set详解

关注:CodingTechWork 创建和分配攻击签名集 可以通过两种方式创建攻击签名集:使用过滤器或手动选择要包含的签名。  基于过滤器的签名集仅基于在签名过滤器中定义的标准。基于过滤器的签名集的优点在于,可以专注于定义用户感兴趣的攻击签名…

@RestControllerAdvice 的作用

系列博客目录 文章目录 系列博客目录1.ControllerAdvice 有什么用主要功能 2.与 RestControllerAdvice 的区别3.苍穹外卖中的使用4.RestControllerAdvice可以指定范围吗(1)指定应用到某些包中的 RestController(2)指定应用到具有特…

代码随想录算法训练营第三十八天-动态规划-完全背包-279.完全平方数

把目标值当作背包容量,每个平方数当作物品,题目变更为装满指定容量的背包,最小用几个物品会不会出现拼凑不出来的情况?不会,因为有数字1,对任意正整数百分百能拼凑出来因此此题目与上一道题就变得一模一样了…

Python帝王學集成-母稿

引用:【【全748集】这绝对是2024最全最细的Python全套教学视频,七天看完编程技术猛涨!别再走弯路了,从零基础小白到Python全栈这一套就够了!-哔哩哔哩】 https://b23.tv/lHPI3XV 语法基础 Python解释器与pycharm编辑器安装 - 定义:Python解释器负责将Python代码转换为计…

springboot 动态配置定时任务

要在Spring Boot中动态配置定时任务,可以使用ScheduledTaskRegistrar类来实现。 首先,创建一个定时任务类,该类需要实现Runnable接口。例如: Component public class MyTask implements Runnable {Overridepublic void run() {/…