当前位置: 首页 > news >正文

【Java0基础学Java第八颗】 -- 继承与多态 -- 多态

8.继承与多态

  • 8.2 多态
    • 8.2.1 多态的概念
    • 8.2.2 多态实现条件
    • 8.2.3 重写
    • 8.2.4 向上转型和向下转型
    • 8.2.5 向下转型
    • 8.2.6 多态的优缺点
    • 8.2.7 避免在构造方法中调用重写的方法

8.2 多态

8.2.1 多态的概念

通俗来说就是多种形态,具体点就是去完成某个行为,当不同的对象去完成时会产生出不同的状态
就比如打印机,彩色打印机和黑白打印机,打印出的效果一个是彩色,一个是黑白。

即:同一件事情,发生在不同对象身上,就会产生不同的结果

8.2.2 多态实现条件

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

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

多态的体现:在代码运行时,当传递不同类对象时,会调用对应类中的方法。

8.2.3 重写

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

  1. 方法名一样
  2. 参数列表相同(个数、数据类型、顺序)
  3. 返回值一样

注意: 不能被重写的4种情况

  1. 被final修饰的方法 不可以被重写。这个方法叫做密封方法。
    public final void eat() {System.out.println(this.name+" 正在吃。。。");}
  1. 被static修饰的方法 不能被重写
public static void eat() {System.out.println(this.name+" 正在吃。。。");}
  1. 子类重写父类方法的时候,子类的方法访问修饰限定符要>=父类
    在这里插入图片描述
// 父类的方法  
//当父类的方法修饰限定符是 默认defau时,子类的方法访问修饰限定符可以是protected、public
public void eat() {System.out.println(this.name+" 正在吃。。。");}//子类的方法 
public void eat() {System.out.println(this.name+" 正在吃狗粮...");}
    1. 被private修饰的方法 是不被重写的

重写和重载的区别
重写的方法名和参数都一样,而重载的方法名相同,参数不同
即:方法重载是一个类的多态性表现,而方法重写是子类与父类的一种多态性表现。

通过反汇编代码:
在这里插入图片描述

动态绑定:子类和父类都有 编译得时候认为 还是确认调用了父类得eat方法
运行得时候 绑定到子类当中

  • 父类引用 引用 子类对象;
  • 通过父类引用 去调用 父类 和子类 同名的方法。

静态绑定:编译的时候就已经确定调用哪个方法了,如下dog.barks(10);
在这里插入图片描述

8.2.4 向上转型和向下转型

向上转型:实际就是创建一个子类对象,将其当成父类对象来使用。
语法格式:父类类型 对象名 = new 子类类型()

  1. 直接赋值
Animal animal = new Dog("旺财",10,"红色"); //直接赋值
  1. 方法传参
	// 方法传参 :形参作为父类引用,可以接受任意子类的对象public static void fun(Animal animal) {animal.eat();// 发生了多态}public static void main(String[] args) {Dog dog = new Dog("旺财",3,"红色");fun(dog);Bird bird = new Bird("布谷",1);fun(bird);}

在这里插入图片描述

同一个引用 调用了 同一个方法,当是因为引用的对象不一样,所表现的行为不一样,称为多态。

  1. 方法返回
	//  通过返回值,进行向上转型public static Animal fun2() {return new Dog("旺财",3,"红色");}

向下转型能让代码变得更加简单灵活,缺点就是不能调用到子类特有的方法。

8.2.5 向下转型

Java中为了提高向下转型的安全性,引入了instanceof ,如果该表达式为true,则可以安全转换。

    public static void main(String[] args) {Animal animal2 = new Bird("布谷",1);//animal2.eat();//animal2.fly();Bird bird = (Bird) animal2;bird.fly();Animal animal1 = new Dog("旺财",3,"红色");if (animal1 instanceof Bird) {Bird bird2 = (Bird) animal1;bird2.fly();}else {System.out.println("不一定所以的动物都是鸟");}}

8.2.6 多态的优缺点

使用多态的好处:

  1. 能够降低代码的圈复杂度,避免使用大量的if-else
    用if-else:
class Shape {public void draw() {System.out.println("画图形");}
}
class Cycle extends Shape {@Overridepublic void draw() {System.out.println("⚪");}
}
class Rect extends Shape {@Overridepublic void draw() {System.out.println("矩形");}
}class Triangle extends Shape {@Overridepublic void draw() {System.out.println("三角形");}
}
public class Test {public static void main(String[] args) {Rect rect = new Rect();Cycle cycle = new Cycle();Triangle triangle = new Triangle();String[] shapes = {"cycle", "rect", "cycle", "rect", "triangle"};for (String shape : shapes) {if (shape.equals("cycle")) {cycle.draw();} else if (shape.equals("rect")) {rect.draw();} else if (shape.equals("triangle")) {triangle.draw();}}}}

用多态:

class Shape {public void draw() {System.out.println("画图形");}
}
class Cycle extends Shape {@Overridepublic void draw() {System.out.println("⚪");}
}
class Rect extends Shape {@Overridepublic void draw() {System.out.println("矩形");}
}class Triangle extends Shape {@Overridepublic void draw() {System.out.println("三角形");}
}
public class Test {// 使用多态public static void main(String[] args) {Rect rect = new Rect();Cycle cycle = new Cycle();Triangle triangle = new Triangle();// 向上转型Shape[] shapes = {cycle,rect,cycle,rect,triangle};for (Shape shape : shapes) {shape.draw();}}
}
  1. 可扩展能力更强
    如果需要新加一种形状,使用多态的方式代码改动成本也比较低
    比如新加一个画❀
class Flower extends Shape {@Overridepublic void draw() {System.out.println("❀");}
}
public class Test {public static void main(String[] args) {Rect rect = new Rect(); // 相当于Shape rect = new Rect();Cycle cycle = new Cycle();Triangle triangle = new Triangle();Flower flower = new Flower();// 向上转型Shape[] shapes = {cycle,rect,cycle,rect,triangle,flower};for (Shape shape : shapes) {shape.draw();}}
}

缺点: 代码的运行效率降低

  1. 属性没有多态性
    当父类和子类都有同名属性的时候,通过父类引用,只能引用父类自己的成员属性
  2. 构造方法没有多态性

8.2.7 避免在构造方法中调用重写的方法

当在父类的构造方法当中,调用父类和子类同名的方法时候,此时也会发生动态绑定。也意味着 构造方法内 也会发生动态绑定。
注意: 在构造方法当中 不要调用重写的方法

class B {public B() {// 父类的实例 --> 父类的构造 -->  子类的实例  --> 子类的构造func();  // 调用子类的func方法  没有给num赋值 所以num的值为0}public void func() {System.out.println("B.func()");}
}
class D extends B {private int num = 1;public D() {super();}@Overridepublic void func() {System.out.println("D.func() " + num);}
}
public class Test2 {public static void main(String[] args) {//1. 分配内存空间  2. 调用合适的构造方法D d = new D();}
}
  • 构造 D 对象的同时, 会调用 B 的构造方法.
  • B 的构造方法中调用了 func 方法, 此时会触发动态绑定, 会调用到 D 中的 func
  • 此时 D 对象自身还没有构造, 此时 num 处在未初始化的状态, 值为 0.。
  • 所以在构造函数内,尽量避免使用实例方法,除了final和private方法。

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

相关文章:

  • 189. 轮转数组
  • leetCode 493 翻转对 归并分治 + 图解
  • Redis系列-四种部署方式-单机部署+主从模式+哨兵模式【7】
  • (层次遍历)104. 二叉树的最大深度
  • pytorch DistributedDataParallel 分布式训练踩坑记录
  • 【问题记录】docker pull 镜像的时候 devel 版本和无 devel 版本的差别
  • 使用 eBPF检测 mmap泄露
  • 【电路笔记】-节点电压分析和网状电流分析
  • EDA实验----四选一多路选择器设计(QuartusII)
  • Java中单例模式
  • Echarts柱状体实现滚动条动态滚动
  • Spring源码系列-框架中的设计模式
  • [工业自动化-11]:西门子S7-15xxx编程 - PLC从站 - 分布式IO从站/从机
  • 【C++笔记】优先级队列priority_queue的模拟实现
  • 原型模式(创建型)
  • 【Leetcode】101. 对称二叉树
  • [Vue CLI 3] 配置解析之 css.extract
  • 【跃迁之路】【669天】程序员高效学习方法论探索系列(实验阶段426-2018.12.13)...
  • Codepen 每日精选(2018-3-25)
  • gf框架之分页模块(五) - 自定义分页
  • Mybatis初体验
  • MyEclipse 8.0 GA 搭建 Struts2 + Spring2 + Hibernate3 (测试)
  • oldjun 检测网站的经验
  • SQLServer之索引简介
  • Webpack4 学习笔记 - 01:webpack的安装和简单配置
  • 从0实现一个tiny react(三)生命周期
  • 面试总结JavaScript篇
  • 深入浏览器事件循环的本质
  • 推荐一个React的管理后台框架
  • 异步
  • LevelDB 入门 —— 全面了解 LevelDB 的功能特性
  • 数据库巡检项
  • 选择阿里云数据库HBase版十大理由
  • ​无人机石油管道巡检方案新亮点:灵活准确又高效
  • $emit传递多个参数_PPC和MIPS指令集下二进制代码中函数参数个数的识别方法
  • (2.2w字)前端单元测试之Jest详解篇
  • (3)选择元素——(17)练习(Exercises)
  • (笔记)Kotlin——Android封装ViewBinding之二 优化
  • (附源码)node.js知识分享网站 毕业设计 202038
  • (三)c52学习之旅-点亮LED灯
  • (三)终结任务
  • (十七)devops持续集成开发——使用jenkins流水线pipeline方式发布一个微服务项目
  • (提供数据集下载)基于大语言模型LangChain与ChatGLM3-6B本地知识库调优:数据集优化、参数调整、Prompt提示词优化实战
  • (转)AS3正则:元子符,元序列,标志,数量表达符
  • (转)关于如何学好游戏3D引擎编程的一些经验
  • (总结)Linux下的暴力密码在线破解工具Hydra详解
  • .net 4.0发布后不能正常显示图片问题
  • .NET 5.0正式发布,有什么功能特性(翻译)
  • .NET 8.0 中有哪些新的变化?
  • .net2005怎么读string形的xml,不是xml文件。
  • .net6解除文件上传限制。Multipart body length limit 16384 exceeded
  • .NET与 java通用的3DES加密解密方法
  • .NET与java的MVC模式(2):struts2核心工作流程与原理
  • /etc/fstab和/etc/mtab的区别
  • [ vulhub漏洞复现篇 ] ECShop 2.x / 3.x SQL注入/远程执行代码漏洞 xianzhi-2017-02-82239600