向上转型 向下转型 重写 多态 ---java

目录

一. 向上转型

1.1 概念

1.2 语法格式 

1.3 动态绑定引入 

1.4 重写的引入 

 1.5向上转型的使用方式

 方式一: 直接赋值

方式二: 通过传参,进行向上转型(多态引入)

方法三:通过返回值, 进行向上转型 

二. 重写

2.1 概念

2.2 重写的格式

2.3 重写的规则

【重写和重载的区别】

 【重写的设计原则】

三. 多态

3.1概念

3.2多态实现条件

3.3 多态的优缺点

 四. 向下转型

4.1 概念

4.2 语法格式

4.3 使用方法


一. 向上转型

1.1 概念

实际就是创建一个子类对象,将其当成父类对象来使用。 父类引用接收子类对象. 

1.2 语法格式 

父类类型 对象名 = new 子类类型 () ;
例:
Animal animal = new Dog("旺财",10);
//animal是父类类型,但可以引用一个子类对象,因为是从小范围向大范围的转换。

1.3 动态绑定引入 

 看下述代码:

看到: animal不能访问barks(), 因为在Animal类中没有.

结论:父类引用只能调用父类自己的方法,不能调用子类的.

接下来, dog类中也定义一个eat()方法:

看到:调用了Dog类中的eat()方法

结论:当父类和子类都有eat()方法时,此时通过animal调用eat()方法,调用的是子类中的方法. 

此时发生的动作叫做动态绑定.

动态绑定:编译的时候, 调用父类的eat方法  运行的时候, 绑定到了子类的eat方法中

静态绑定:编译的时候 已经确定调用哪个方法

1.4 重写的引入 

 拿出父类和子类中的eat()进行对比:

发现:

1. 方法名相同

2. 参数列表相同(顺序, 个数, 类型)

3. 返回值相同

 此时发生了重写.(后面讲)二. 重写

 1.5向上转型的使用方式

对上述代码再添加一个Bird类

 方式一: 直接赋值

   子类对象赋值给父类对象

方式二: 通过传参,进行向上转型(多态引入)

   形参为父类型引用,可以接收任意子类的对象

当我们在方法中加入animal.eat(); 如下:

 

看到:

同一个引用 调用了同一个方法

但因为发生了向上转型 引用的对象不一样 所表现的行为不一样

--->我们把这种思想叫做多态.(后面讲)三. 多态

方法三:通过返回值, 进行向上转型 

返回任意子类对象

二. 重写

2.1 概念

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

2.2 重写的格式

1. 方法名相同

2. 参数列表相同(顺序, 个数, 类型)

3. 返回值相同

2.3 重写的规则

1. 当父类中的方法被 final 修饰时, 此方法不能被重写. 这个方法叫做密封方法.

2. 可以在子类方法前加入 @Override 注解来显式指定, 表示此方法是重写方法, 有了这个注解能帮我们进行一些合法性校验.

3. 当父类的方法被 static 修饰时,  此方法不能被重写. 

4.  子类重写方法的访问权限不能比父类中被重写的方法的访问权限更低。

访问修饰限定符:子类 >= 父类

例如:如果父类方法被public修饰,则子类中重写该方法就不能声明为 protected.

5. 当父类中的方法被 private 修饰时, 此方法不能被重写.(private只能在同一类中使用)

6. 父类与子类返回值类型可以不同,但是必须是具有父子关系的  

父类:

子类:

 

 父类与子类返回值类型可以不同,但构成了父子关系, 也是可以的.

*7. 在构造方法中, 避免调用重写的方法. 

例如下述代码:

结果: 

 

看到:

当在父类的构造方法当中,调用父类和子类同名的方法的时候, 此时也会发生动态绑定, 运行时,绑定到了子类的方法中. 

也意味着, 构造方法内, 也会发生动态绑定.

 但为什么num值为0呢?

在之前的学习中, 我们了解到实例化一个对象之后, 简单来说会发生两件事情:

1. 分配内存空间

2. 调用合适的构造方法

所以在new D()之后,为d这个对象分配了一块空间, 也就是为num也分配了空间, 在调用构造方法时, 我们学过顺序:

1. 父类的实例

2. 父类的构造

3. 子类的实例

4. 子类的构造

所以我们首先调用了父类的构造方法, 在方法中调用了子类的func(), 打印了num, 此时num默认值为0, 只有当执行到子类的实例这一步时, num才被初始化为1.

执行顺序如下图:

重写和重载的区别

 

 重写的设计原则

对于已经投入使用的类,尽量不要进行修改。最好的方式是:重新定义一个新的类,来重复利用其中共性的内容, 并且添加或者改动新的内容。
例如:若干年前的手机,  只能打电话,发短信,来电显示只能显示号码,而今天的手机在来电显示的时候,不仅仅 可以显示号码,还可以显示头像,地区等。在这个过程当中,我们不应该在原来老的类上进行修改,因为原来的 类,可能还在有用户使用 ,正确做法是: 新建一个新手机的类,对来电显示这个方法重写就好了,这样就达到了我 们当今的需求了

三. 多态

3.1概念

通俗来说,就是多种形态, 具体点就是去完成某个行为,当不同的对象去完成时会产生出不同的状 态。

3.2多态实现条件

java 中要实现多态,必须要满足如下几个条件,缺一不可:
1. 必须在继承体系下
2. 子类必须要对父类中方法进行重写
3. 通过父类的引用调用重写的方法

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

例:

class Animal{String name;int age;public Animal(String name, int age) {this.name = name;this.age = age;}public void eat(){System.out.println(this.name +"正在吃...");}
}class Dog extends Animal{public Dog(String name, int age) {super(name, age);}public void eat(){System.out.println(this.name +"正在吃狗粮...");}void bark(){System.out.println(name + "汪汪汪~~~");}
}class Bird extends Animal{public Bird(String name, int age) {super(name, age);}public void fly(){System.out.println(this.name+"正在飞...");}public void eat(){System.out.println(this.name+"正在吃鸟粮...");}
}
//分割线
public class test {// 注意:此处的形参类型必须是父类类型才可以public static void fun(Animal animal){animal.eat();// 编译器在编译代码时,并不知道要调用Dog 还是 Bird 中eat的方法// 等程序运行起来后,形参animal引用的具体对象确定后,才知道调用那个方法}public static void main(String[] args) {Dog dog = new Dog("旺财",10);fun(dog);Bird bird = new Bird("布谷",1);fun(bird);}
}

 

在上述代码中 , 分割线上方的代码是 类的实现者 编写的 , 分割线下方的代码是 类的调用者 编写的 .
当类的调用者在编写 eat 这个方法的时候 , 参数类型为 Animal ( 父类 ), 此时在该方法内部并 不知道 , 也不关注 当前的 animal 引用指向的是哪个类型 ( 哪个子类 ) 的实例 . 此时  animal  这个引用调用 eat  方法可能会有多种不同的表现 (  animal  引用的实例相关), 这种行为就称为 多态 .

3.3 多态的优缺点

例下述代码: 

class Shape {
        //属性 ....
        public void draw () {
                System . out . println ( " 画图形! " );
        }
}
class Rect extends Shape {
        @Override
        public void draw () {
                System . out . println ( " " );
        }
}
class Cycle extends Shape {
        @Override
        public void draw () {
                System . out . println ( " " );
        }
}
class Flower extends Shape {
        @Override
        public void draw () {
                System . out . println ( " " );
        }
}

我们现在需要打印多个形状. 如果不基于多态, 实现代码如下:  

public static void drawShapes () {
        Rect rect = new Rect ();
        Cycle cycle = new Cycle ();
        Flower flower = new Flower ();
        String [] shapes = { "cycle" , "rect" , "cycle" , "rect" , "flower" };
        for ( String shape : shapes ) {
                if ( shape . equals ( "cycle" )) {
                        cycle . draw ();
                } else if ( shape . equals ( "rect" )) {
                        rect . draw ();
                } else if ( shape . equals ( "flower" )) {
                        flower . draw ();
                }
        }
}

如果使用使用多态, 则不必写这么多的 if - else 分支语句, 代码更简单.

public static void drawShapes () {
// 我们创建了一个 Shape 对象的数组 .
        Shape [] shapes = { new Cycle (), new Rect (), new Cycle (), new Rect (), new Flower ()};
//向上转型
        for ( Shape shape : shapes ) {
                shape . draw ();
        }
}
使用多态的好处
1. 能够降低代码的 " 圈复杂度 ", 避免使用大量的 if - else
什么叫 " 圈复杂度 " ?
圈复杂度是一种描述一段代码复杂程度的方式 . 一段代码如果平铺直叙 , 那么就比较简单容易理解 . 而如果有很多的条件分支或者循环语句, 就认为理解起来更复杂 .
因此我们可以简单粗暴的计算一段代码中条件语句和循环语句出现的个数 , 这个个数就称为 " 圈复杂度 ". 如果一个方法的圈复杂度太高, 就需要考虑重构 .
不同公司对于代码的圈复杂度的规范不一样 . 一般不会超过 10.
2. 可扩展能力更强
如果要新增一种新的形状 , 使用多态的方式代码改动成本也比较低 .
多态缺陷:代码的运行效率降低
1. 属性没有多态性
当父类和子类都有同名属性的时候,通过父类引用,只能引用父类自己的成员属性
2. 构造方法没有多态性

 四. 向下转型

4.1 概念

将一个子类对象经过向上转型之后当成父类方法使用,再无法调用子类的方法,但有时候可能需要调用子类特有的方法,此时:将父类引用再还原为子类对象即可,即向下转型。

4.2 语法格式

父类类型 对象名 = new 子类类型() ;//向上转型

子类类型 对象名 = (子类类型) 父类对象名 ;//向下转型

4.3 使用方法

1.

结果: 

2. 

 

结果: 

程序可以通过编程,但运行时抛出异常---因为:animal2实际指向的是鸟, 不能强转成狗

所以:

向下转型用的比较少,而且不安全,万一转换失败,运行时就会抛异常。
Java 中为了提高向下转型的安全性,引入了 instanceof ,判断这个对象是否属于这个类型, 如果该表达式为 true ,则可以安全转换。

结果: 

 强转失败

今天的分享就到这里, 谢谢大家的点赞支持! 

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

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

相关文章

探索分销小程序商城开发和搭建

在数字化时代,小程序已经成为了企业营销的重要工具。其中,分销小程序商城以其独特的优势,吸引了众多企业的关注。下面给大家介绍如何开发分销小程序商城,以及小程序分销搭建的流程。 首先,我们需要明确什么是分销小程…

A股风格因子看板 (2023.11第12期)

该因子看板跟踪A股风格因子,该因子主要解释沪深两市的市场收益、刻画市场风格趋势的系列风格因子,用以分析市场风格切换、组合风格暴露等。 今日为该因子跟踪第12期,指数组合数据截止日2023-10-31,要点如下: 近1年A股风…

RK3568开发板在工控工业物联网网关方面的应用

在数字化转型的浪潮中,工控物联网关产品扮演着重要的角色。这些产品通过连接工业设备和网络,为数据传输和分析提供了便利。而迅为RK3568核心板作为一款高性能的芯片,为工控物联网关产品的性能提升和功能扩展提供了强大的支持。 迅为RK3568核心…

3.10-容器的操作

这一节讲解一下对于container我们可以进行哪些操作&#xff1f; 可以使用以下命令来停止正在运行的Docker容器&#xff1a; docker container stop <CONTAINER ID> 关于运行中的容器&#xff0c;我们可以进行的操作&#xff1a; 第一个是docker exec命令&#xff0c;这个…

用Markdown Nice写作

网址&#xff1a;https://www.mdnice.com/ 代码 表格 第二行用来对齐&#xff1a; -表示左对齐 :-:表示居中 -:表示右对齐 数学 上下标 分数 累加 幂 对数 根式 微积分 交集、并集 格式 标题 缩进 删除线 斜体 加粗 参考文献

NX二次开发UF_CAM_PREF_set_logical_value 函数介绍

文章作者&#xff1a;里海 来源网站&#xff1a;https://blog.csdn.net/WangPaiFeiXingYuan UF_CAM_PREF_set_logical_value Defined in: uf_cam_prefs.h int UF_CAM_PREF_set_logical_value(UF_CAM_PREF_t pref, logical value ) overview 概述 This function sets the lo…

Linux 局域网传输工具LANDrop安装

Linux 局域网传输工具LANDrop安装 &#x1f959;下载&#x1f32d;解压&#x1f96a;运行 &#x1f959;下载 官网下载 或网盘 &#x1f32d;解压 使用以下命令解压获得squashfs-root文件夹 ./LANDrop-latest-linux.AppImage --appimage-extract&#x1f96a;运行 进入squ…

YOLOv5 环境搭建

YOLOv5 环境搭建 flyfish 环境 Ubuntu20.04 驱动、CUDA Toolkit、cuDNN、PyTorch版本对应 1 NVIDIA驱动安装 在[附加驱动界]面安装驱动时&#xff0c;需要输入安全密码&#xff0c;需要记下&#xff0c;后面还需要输入这个密码 重启之后有的机器会出现 perform mok manage…

生产订单自动下达

文章目录 1 Introduction2 Detail2.1 input MM02 3 Summary 1 Introduction Production order is released order by automation . We can set the material for it . The method is the detial . 2 Detail 2.1 input MM02 please choose work Scheduling please choose S…

2023年约特干故城夜间演艺《万方乐奏有于阗》完美谢幕

11月19日&#xff0c;记者走进约特干故城看到演员在欢乐地跳着刀郎舞和古典舞&#xff0c;庆祝今年以来夜间演艺《万方乐奏有于阗》演出200场完美谢幕。 11月19日在约特干故城&#xff0c;演员正在表演迎宾乐舞。阿卜力克木依卜拉依木摄 当天晚上&#xff0c;城楼上旌旗猎猎&am…

【css】Google第三方登录按钮样式修改

文章目录 场景前置准备修改样式官方属性修改样式CSS修改样式按钮的高度height和border-radiusLogo和文字布局 场景 需要用到谷歌的第三方登录&#xff0c;登录按钮有自己的样式。根据官方文档&#xff1a;概览 | Authentication | Google for Developers&#xff0c;提供两种第…

满二叉树你需要了解一下

满二叉树介绍 满二叉树&#xff08;Full Binary Tree&#xff09;是一种特殊的二叉树&#xff0c;其中每个节点都有两个子节点或没有子节点。换句话说&#xff0c;满二叉树的每个层级都是完全填满的。这种树结构具有一定的平衡性&#xff0c;其深度和节点数量之间存在明确的关…