Java 多态

news/2024/11/20 4:33:33/

文章目录

    • 1、多态的介绍
    • 2、多态的格式
    • 3、对象的强制类型转换
    • 4、instanceof 运算符
    • 5、案例:笔记本USB接口

1、多态的介绍

多态(Polymorphism)按字面意思理解就是“多种形态”,即一个对象拥有多种形态。

即同一种方法可以根据发送对象的不同而采用多种不同的行为方式。

一个对象的实际类型是确定的,但可以指向对象的引用的类型有很多。

多态存在的条件

  • 有继承(基类)或 实现 (接口) 关系
  • 子类重写父类的方法
  • 父类引用变量指向子类对象

==注:==多态是针对实例方法的多态,不是属性的多态,也不是类方法的多态。

java 引用变量分为两种类型,一种是编译时类型,一种是运行时类型。

编译时类型由声明该变量时使用的类型决定;运行时类型由实际赋值给该变量的对象决定。

如果编译时类型和运行时类型不一致,就**可能(重写了方法的话)**出现所谓的多态。

class Person {public void eat() {	}
}class Student extends Person {// 重写了基类中的方法public void eat() {}// 子类中特有的方法public void study() {}
}class Applicaton {public static void main(String[] args) {// 不存在多态 编译时类型和运行时类型一致Student s3 = new Student();// 不存在多态 编译时类型和运行时类型一致Person s2 = new Person();// 存在多态 编译时类型和运行时类型不一致,重写了基类中的方法Person s1 = new Student();s1.eat();// 报错 原因下方会解释// s1.study();}
}

当把一个子类对象直接赋值给父类引用变量时:

例如上面的 Person s1 = new Student();

这个 s1 的引用变量的编译类型是 Person,而运行时类型是 Student,当运行时调用该引用变量的方法时,其方法行为总是表现出子类方法的行为特征,而不是父类方法的行为特征,这就可能出现:相同类型的变量调用同一个方法时呈现出多种不同的行为特征,这就是多态。

这里的相同类型就是指同一个引用变量(s1),调用同一个方法,但指向不同的子类对象时,表现出不同的行为特征。

注:

1、通过引用变量来直接访问其包含的实例变量时,系统总是试图访问它编译时类型定义的实例变量,而不是运行时类型所定义的实例变量。

2、引用变量在编译阶段只能调用其编译时类型所具有的方法,但运行阶段则可以调用它运行时类型所具有的的方法。

还是上面的例子:Person s1 = new Student(); s1 引用变量只能调用 Person 类的方法,不能直接调用 Student 里独有的方法。

作用

简单来说,就是建立一个父类对象的引用变量,指向不同的子类对象。

把不同的子类对象都当做父类来看待,屏蔽不同子类对象之间的差异,写出通用的代码,以适应需求的不断变化。

而且无论右边 new 的时候换成了哪个子类对象,等号左边的调用方法都不会变化。

2、多态的格式

  • 代码当中体现多态性,其实就是一句话:父类引用指向子类对象
  • 格式:
基类类型 引用变量名称 = new 子类类型(); // 通过继承实现
or
接口类型 引用变量名称 = new 实现类类型(); // 通过接口实现

对象的自动类型

多态的写法其实就是对象的自动类型(类型向上转),只不过是编译器自动完成的。

注:

向上转型一定是成功的,这也证实了子类其实是一种特殊的父类。

这种类型转换只是表明这个引用变量的编译时类型是基类,运行时类型是子类,执行方法时表现的是子类对象的行为方式。

如果想让基类引用变量调用子类中特有的方法,就需要进行强制类型转换。

3、对象的强制类型转换

强制类型转换需要借助于类型转换运算符—— (类型)

类型转换运算符的用法:(类型)变量名称

通过类型转换运算符将一个父类引用变量强转成子类引用变量时,需要注意一下几点:

1、基本类型之间的转换只能在数值类型之间进行。

数值类型:整型(byte、short、int、long)、字符型(char)、浮点型(float、double)。

数值类型和布尔类型之间不能进行类型转换。

2、引用类型之间的转换只能在具有继承 or 实现关系的两个类型之间进行,如果把两个不相关的类型之间进行转换,则会报类型转换异常:ClassCastException

示例

Person s1 = new Student();
// Person 类中没有 study() 方法 进行强制类型转换
((Student) s1).study();

考虑到进行强制类型转换时可能出现的异常,因此进行类型转换之前应先通过 instanceof 运算符来判断是否可以成功转换,避免出现异常,以增强程序的健壮性。

4、instanceof 运算符

使用:A instanceof B

作用:判断其左边对象(指运行时对象)是否为其右边类的实例,返回 boolean 类型的数据。

如果 A 和 B 不存继承话,会出现编译错误。

示例 1:

class Animal {}class Cat extends Animal {}class Dog extends Animal  {}class Applicaton {public static void main(String[] args) {Animal animal = new Dog();if (animal instanceof Dog) {System.out.println("汪汪~");} else if (animal instanceof Cat) {System.out.println("喵喵~");}}
}

实例 2:

// obj引用变量 编译时类型是 Object,运行时类型是String
// 因为Object是所有类、接口的父类,所以以下才可通过
Object obj = "chatGPT";
// String 是 Object 类的实例
System.out.println(obj instanceof Object);
// String 是 String 类的实例
System.out.println(obj instanceof String);
// String 不是 Math 类的实例
System.out.println(obj instanceof Math);
// String 是 Comparable 接口的实例。接口理解成一种特殊的类
System.out.println(obj instanceof Comparable);// 注意:这里的 str 没有用多态的写法
String str = "hello";// 编译不通过,str引用变量编译时类型是String和Math没有继承关系
// System.out.println(str instanceof Math);// String 对象是 Object 类的实例
System.out.println(str instanceof Object);

5、案例:笔记本USB接口

重点:接口的基本使用、对象的上下转型以及使用接口作为方法的参数。
Computer 使用 USB接口,Mouse、KeyBoards 实现 USB接口

USB 接口

// USB接口类
public interface USB {void open();void close();
}

Computer 类

// 电脑类
public class Computer {public void powerOn(){System.out.println("电脑开机");}public void powerOff(){System.out.println("电脑关机");}// 接口作为方法的参数public void useDevice(USB usb){usb.open();if (usb instanceof Mouse){ // 一定要先判断Mouse mouse = (Mouse)usb; // 向下转型mouse.click();} else if (usb instanceof KeyBoard) {KeyBoard keyboard = (KeyBoard)usb;keyboard.type();}usb.close();}
}

Mouse 类

// 鼠标类
public class Mouse implements USB{@Overridepublic void open() {System.out.println("打开鼠标");}@Overridepublic void close() {System.out.println("关闭鼠标");}public void click(){System.out.println("鼠标点击");}
}

KeyBoard 类

// 键盘类
public class KeyBoard implements USB{@Overridepublic void open() {System.out.println("打开键盘");}@Overridepublic void close() {System.out.println("关闭键盘");}public void type(){System.out.println("键盘输入");}
}

Demo02Main 测试类

public class Demo02Main {public static void main(String[] args) {Computer computer = new Computer();computer.powerOn();USB usbMouse = new Mouse();computer.useDevice(usbMouse);//        USB usbKeyBoard = new KeyBoard();KeyBoard keyBoard = new KeyBoard();computer.useDevice(keyBoard);computer.powerOff();}
}

http://www.ppmy.cn/news/24189.html

相关文章

linux 线程

文章目录1、线程的概念1.1、进程 vs 线程1.2、线程的种类2、线程的控制2.1、线程的创建2.2、线程的退出2.3、线程的取消2.4、线程的等待2.5、线程的分离2.5、线程清理函数线程清理函数响应的时机线程清理函数不响应的时机3、线程的同步和互斥3.1、锁机制3.1.1、锁的类型3.1.2、…

求职(JAVA程序员的面试自我介绍)

背景 在找工作的过程中,在面试的环节,大多数面试官首先都会叫你自我介绍一下。一般是3到5分钟内。不过经过我面试的无数的公司还有曾经也面试过大多数的求职者。国内很多的程序员面试都极其不专业。有一种很随心所欲的感觉。所以经常遇到求职者吐槽遇到了…

02- 天池工业蒸汽量项目实战 (项目二)

忽略警告: warnings.filterwarnings("ignore") import warnings warnings.filterwarnings("ignore") 读取文件格式: pd.read_csv(train_data_file, sep\t) # 注意sep 是 , , 还是\ttrain_data.info() # 查看是否存在空数据及数据类型train_data.desc…

Linux内核启动(3,0.11版本)内核启动完成与进入内核main函数

这一部分是在讲解head.s代码,这个代码与bootsect.s和setup.s在同一目录下,但是head.s程序在被编译生成目标文件后会与内核其他程序一起被链接成system模块,位于system模块的最前面开始部分。system模块将被放置在磁盘上setup模块之后开始的扇…

d3.js与echarts对比

D3.js 和 ECharts 是两种常用的数据可视化工具,它们有着不同的优缺点: D3.js: 优点: 功能强大,提供了极高的灵活性和定制性,支持多种图表类型,如柱状图、饼图、散点图、树图、网络图等。 可以…

LSTM已死,Transformer当立(LSTM is dead. Long Live Transformers! ):下

2017 年,Google 在论文 Attention is All you need 中提出了 Transformer 模型,其使用 Self-Attention 结构取代了在 NLP 任务中常用的 RNN 网络结构。而且实验也证明Transformer 在效果上已经完败传统的 RNN 网络。Transformer 的整体模型架构如下图所示。尽管它看起来还是很…

【Hello Linux】 Linux基础命令

作者:小萌新 专栏:Linux 作者简介:大二学生 希望能和大家一起进步! 本篇博客简介:介绍Linux的基础命令 Linux基础命令ls指令lsls -als -dls -ils -sls -lls -nls -Fls -rls -tls -Rls -1总结思维导图pwd指令whoami指令…

初学者必读:讲解 VC 下如何正确的创建、管理及发布项目

Visual C 的项目文件组成,以及如何正确的创建及管理项目。 本内容是初学者必须要掌握的。不能正确的管理项目,就不能进一步写有规模的程序。 一、项目下各种常见文件类型的功能 1. 代码文件 扩展名为 .cpp、.c、.h 等。 通常情况下,项目…