草庐IT

Java多态详解

无贪则喜 2023-04-19 原文

目录

1. 基本介绍

1.1 多态的概念

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

1.2 多态的具体体现

对象的多态是多态的核心和重点

规则

  • 一个对象的编译类型与运行类型可以不一致
  • 编译类型在定义对象时,就确定了,不能改变,而运行类型是可以变化的
  • 编译类型看定义对象时 = 号的左边运行类型看 = 号的右边

1.3 入门案例

说明

  • 定义一个 Person 类 作为父类,定义 Student 类 和 Teacher 类作为子类继承父类
  • Person 类 拥有 mission() 方法;
  • Student 类 和 Teacher 类 重写父类的 mission() 方法;
  • 最后在 main 函数中利用多态形式创建对象。

代码如下

(1)定义父类 Person 类:

package Polymorphism;

public class Person {
	public void mission() {	
		System.out.println("人要好好活着!");
	}
}

(2)定义子类 Student 类:

package Polymorphism;

public class Student extends Person {
	@Override
	public void mission() {	
		System.out.println("学生要好好学习!");
	}
}

(3)定义子类 Teacher 类

package Polymorphism;

public class Teacher extends Person {
	@Override
	public void mission() {	
		System.out.println("老师要好好教书!");
	}
}

(4)在 Test01 类中编写 main 函数,体现多态性

package Polymorphism;

//多态性
public class Test01 {
	public static void main(String[] args) {
		//多态形式,创建对象
		//注意编译类型看等号左边,运行类型看等号右边
		Person p1 = new Student();  
		
		//此时调用的是 Student 类 的 mission() 方法
		p1.mission();
		
		//多态形式,创建对象
		Person p2 = new Teacher();
		
		//此时调用的是 Teacher 类 的 mission() 方法
		p2.mission();
	}
}

(5)运行结果

学生要好好学习!
老师要好好教书!

2. 多态的转型

2.1 向上转型

  1. 本质:父类的引用指向子类的对象

  2. 特点

  • 编译类型看左边,运行类型看右边
  • 可以调用父类的所有成员(需遵守访问权限)
  • 不能调用子类的特有成员
  • 运行效果看子类的具体实现
  1. 语法
父类类型 引用名 = new 子类类型();
//右侧创建一个子类对象,把它当作父类看待使用

2.2 向下转型

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

  2. 特点

  • 只能强制转换父类的引用,不能强制转换父类的对象
  • 要求父类的引用必须指向的是当前目标类型的对象
  • 当向下转型后,可以调用子类类型中所有的成员
  1. 语法
子类类型 引用名 = (子类类型) 父类引用;
//用强制类型转换的格式,将父类引用类型转为子类引用类型

2.3 代码示例

说明

  • 定义一个 Person 类 作为父类,定义 Student 类 和 Teacher 类作为子类继承父类;
  • Person 类 拥有 mission() 方法;
  • Student 类 和 Teacher 类 重写父类的 mission() 方法 并且 分别拥有各自的特有的 score() 方法 和 salary() 方法;
  • 最后在 main 函数中 演示转型

代码如下

(1)定义类:

package Poly_;

public class Person {
	public void mission() {	
		System.out.println("人要好好活着!");
	}
}

class Student extends Person {	
	@Override
	public void mission() {	
		System.out.println("学生要好好学习!");
	}
	
	public void score() {
		System.out.println("学生得到好成绩!");
	}
}

class Teacher extends Person {
	@Override
	public void mission() {	
		System.out.println("老师要好好教书!");
	}
	
	public void salary() {	
		System.out.println("老师得到高工资!");
	}
}

(2)在 Test02 类中编写 main 函数,演示转型

package Poly_;

//转型演示
public class Test02 {
	public static void main(String[] args) {
		//向上转型(自动类型转换)
		Person p1 = new Student();
		
		//调用的是 Student 的 mission
		p1.mission(); 
		
		//向下转型
		Student s1 = (Student)p1;
		
		//调用的是 Student 的 score
		s1.score();
	}
}

(3)运行结果:

学生要好好学习!
学生得到好成绩!

2.4 转型的异常

2.4.1 类型转换异常

说明:使用强转时,可能出现异常,对2.3代码示例中的 Test02类 重新编写,演示转型异常

代码如下

//异常演示
public class Test02 {
	public static void main(String[] args) {
		//向上转型
		Person p1 = new Student();
		
		//调用的是 Student 的 mission
		p1.mission(); 
		
		//向下转型
		Teacher t1 = (Teacher) p1;
		
		//运行时报错
		p1.salary();
	}
}

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

2.4.2 instanceof 比较操作符

为了避免上述类型转换异常的问题,我们引出 instanceof 比较操作符,用于判断对象的类型是否为XX类型或XX类型的子类型

  • 格式:对象 instanceof 类名称
  • 解释:这将会得到一个boolean值结果,也就是判断前面的对象能不能当作后面类型的实例
  • 代码示例
package Poly_;

//演示 instanceof 的使用
public class Test03 {
	public static void main(String[] args) {
		//向上转型
		Person p1 = new Student();
		
		//调用的是 Student 的 mission
		p1.mission();
		
		//向下转型
		//利用 instanceof 进行判断
		if(p1 instanceof Student) {	//判断对象 p1 是否是 Student 类 的实例
			Student s1 = (Student)p1;
			s1.score();  //调用的是 Student 的 score
			//上面这两句也可简写为 ((Student) p1).score();
		}
		else if(p1 instanceof Teacher){ //判断对象 p1 是否是 Teacher 类 的实例
			Teacher t1 = (Teacher)p1;
			t1.salary(); //调用的是 Teacher 的 salary
			//同理,上面这两句也可简写为 ((Teacher) p1).salary();
		}
	}
}
  • 运行结果:
学生要好好学习!
学生得到好成绩!

3.动态绑定(重点)

  • 当调用对象方法的时候,该方法会和该对象的运行类型绑定
  • 当调用对象属性时,没有动态绑定机制,即哪里声明,哪里使用。
  • 代码示例
package dynamic_;

//演示动态绑定
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();
	}
}

//父类
class Person {
	public void mission() {	
		System.out.println("人要好好活着!");
	}
}

//子类
class Student extends Person {
	@Override
	public void mission() {	
		System.out.println("学生要好好学习!");
	}
}
  • 运行结果:
学生要好好学习!

4. 应用

4.1 多态数组

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

  1. 说明
  • 定义一个 Person 类 作为父类,定义 Student 类 和 Teacher 类 作为子类继承父类
  • Person 类 拥有 name(姓名) 属性 以及 mission() 方法
  • Student 类 和 Teacher 类 拥有各自特有的 score 和 salary 属性,,除此之外,重写父类的 mission() 方法
  • 要求:最后在 main 函数中 创建一个 Person 对象 、一个 Student 对象 和 一个 Teacher 对象,统一放在数组里,并调用每个对象的 mission() 方法。
  1. 代码如下

(1)父类 Person 类:

package polyarr;

public class Person {
	private String name;
	
	public Person(String name) {
		this.name = name;
	}
	
	// getter 和 setter
	public String getName() {
		return name;
	}
	public void setName(String name) {
		this.name = name;
	}

	// mission() 方法
	public String mission() {	
		return name + "\t" + "要好好活着";
	}
}

(2)子类 Student 类

package polyarr;

public class Student extends Person {
	private double score;

	public Student(String name, double score) {
		super(name);
		this.score = score;
	}

	public double getScore() {
		return score;
	}

	public void setScore(double score) {
		this.score = score;
	}
	
	//重写父类的say方法
	@Override
	public String mission() {	
		return super.mission() + " score =" + score + " 要好好学习!";
	}
}

(3)子类 Teacher 类

package polyarr;

public class Teacher extends Person {
	private double salary;

	public Teacher(String name, double salary) {
		super(name);
		this.salary = salary;
	}

	public double getSalary() {
		return salary;
	}

	public void setSalary(double salary) {
		this.salary = salary;
	}
	
	//重写父类的 mission 方法
	@Override
	public String mission() {	
		return super.mission() + " salary =" + salary + " 要好好教书!";
	}
}

(4)PolyArray 类 中编写 main 函数

package polyarr;

/*
 * 演示多态数组
 * 创建一个 Person 对象 
 * 创建一个 Student 对象 
 * 创建一个 Teacher 对象
 * 统一放在数组里,并调用每个对象的 mission() 方法。
 */
public class PolyArray {
	public static void main(String[] args) {
		Person[] persons = new Person[3];
		persons[0] = new Person("小汤");
		persons[1] = new Student("小韬", 100);
		persons[2] = new Teacher("小蒲", 10000);
		
		//循环遍历多态数组,调用 mission
		for(int i = 0; i < persons.length; i++) {
			//此处涉及动态绑定机制
			// Person[i] 编译类型是 Person ,运行类型根据实际情况由 JVM 判断
			System.out.println(persons[i].mission());  
		}
	}
}

(5)运行结果:

小汤	要好好活着!
小韬	要好好活着! score = 100.0 要好好学习!
小蒲	要好好活着! salary = 10000.0 要好好教书!

4.2 多态参数

多态参数:方法定义的形参类型父类类型,实参类型允许为子类类型。
代码示例

  1. 说明
  • 定义一个 Person 类 作为父类,定义 Student 类 和 Teacher 类作为子类继承父类
  • Person 类 拥有 name(姓名) 属性
  • Student 类 和 Teacher 类 拥有各自 特有 的 study() 和 teach() 方法
  • 要求:最后在 main 函数中 编写 test() 方法 ,功能是调用 Student 类 的 study() 或 Teacher 类 的 teach() 方法,用于演示 多态参数 的使用。
  1. 代码如下
package polyparameter;

//演示多态参数
public class PolyParameter { 
	public static void main(String[] args) {
		Student s1 = new Student("小蓝同学");
		Teacher t1 = new Teacher("小绿老师");
		
		//需先 new 一个当前类的实例化,才能调用 test 方法
		PolyParameter polyParameter = new PolyParameter();
		
		//实参是子类
		polyParameter.test(s1);
		polyParameter.test(t1);		
	}

	//定义方法test,形参为 Person 类型(形参是父类)
	//功能:调用学生的study或教师的teach方法
	 public void test(Person p) {
        if (p instanceof Student){
            ((Student) p).study();   //向下转型
        }
        else if (p instanceof Teacher){
            ((Teacher) p).teach();  //向下转型
        }  
	 }
}
 
//父类
class Person {
	private String name;
	
	//有参构造
	public Person(String name) {
		this.name = name;
	}
	
	// getter 和 setter
	public String getName() {
		return name;
	}
	public void setName(String name) {
		this.name = name;
	}
}

//子类
class Student extends Person {

	public Student(String name) {
		super(name);
	}

	// study() 方法
	public void study() {	
		System.out.println(super.getName() + "\t" + "正在好好学习");
	}
}

class Teacher extends Person {

	public Teacher(String name) {
		super(name);
	}

	// teach() 方法
	public void teach() {	
		System.out.println(super.getName() + "\t" + "正在好好教书");
	}
}
  1. 运行结果:
小蓝同学	正在好好学习
小绿老师	正在好好教书

5. 多态的优点

  • 代码更加灵活:无论右边new的时候换成哪个子类对象,等号左边调用方法都不会变化。
  • 提高程序的拓展性:定义方法的时候,使用父类类型作为参数,将来使用时,使用具体的子类类型操作

有关Java多态详解的更多相关文章

  1. java - 等价于 Java 中的 Ruby Hash - 2

    我真的很习惯使用Ruby编写以下代码:my_hash={}my_hash['test']=1Java中对应的数据结构是什么? 最佳答案 HashMapmap=newHashMap();map.put("test",1);我假设? 关于java-等价于Java中的RubyHash,我们在StackOverflow上找到一个类似的问题: https://stackoverflow.com/questions/22737685/

  2. java - 从 JRuby 调用 Java 类的问题 - 2

    我正在尝试使用boilerpipe来自JRuby。我看过guide从JRuby调用Java,并成功地将它与另一个Java包一起使用,但无法弄清楚为什么同样的东西不能用于boilerpipe。我正在尝试基本上从JRuby中执行与此Java等效的操作:URLurl=newURL("http://www.example.com/some-location/index.html");Stringtext=ArticleExtractor.INSTANCE.getText(url);在JRuby中试过这个:require'java'url=java.net.URL.new("http://www

  3. java - 我的模型类或其他类中应该有逻辑吗 - 2

    我只想对我一直在思考的这个问题有其他意见,例如我有classuser_controller和classuserclassUserattr_accessor:name,:usernameendclassUserController//dosomethingaboutanythingaboutusersend问题是我的User类中是否应该有逻辑user=User.newuser.do_something(user1)oritshouldbeuser_controller=UserController.newuser_controller.do_something(user1,user2)我

  4. java - 什么相当于 ruby​​ 的 rack 或 python 的 Java wsgi? - 2

    什么是ruby​​的rack或python的Java的wsgi?还有一个路由库。 最佳答案 来自Python标准PEP333:Bycontrast,althoughJavahasjustasmanywebapplicationframeworksavailable,Java's"servlet"APImakesitpossibleforapplicationswrittenwithanyJavawebapplicationframeworktoruninanywebserverthatsupportstheservletAPI.ht

  5. Observability:从零开始创建 Java 微服务并监控它 (二) - 2

    这篇文章是继上一篇文章“Observability:从零开始创建Java微服务并监控它(一)”的续篇。在上一篇文章中,我们讲述了如何创建一个Javaweb应用,并使用Filebeat来收集应用所生成的日志。在今天的文章中,我来详述如何收集应用的指标,使用APM来监控应用并监督web服务的在线情况。源码可以在地址 https://github.com/liu-xiao-guo/java_observability 进行下载。摄入指标指标被视为可以随时更改的时间点值。当前请求的数量可以改变任何毫秒。你可能有1000个请求的峰值,然后一切都回到一个请求。这也意味着这些指标可能不准确,你还想提取最小/

  6. 【Java 面试合集】HashMap中为什么引入红黑树,而不是AVL树呢 - 2

    HashMap中为什么引入红黑树,而不是AVL树呢1.概述开始学习这个知识点之前我们需要知道,在JDK1.8以及之前,针对HashMap有什么不同。JDK1.7的时候,HashMap的底层实现是数组+链表JDK1.8的时候,HashMap的底层实现是数组+链表+红黑树我们要思考一个问题,为什么要从链表转为红黑树呢。首先先让我们了解下链表有什么不好???2.链表上述的截图其实就是链表的结构,我们来看下链表的增删改查的时间复杂度增:因为链表不是线性结构,所以每次添加的时候,只需要移动一个节点,所以可以理解为复杂度是N(1)删:算法时间复杂度跟增保持一致查:既然是非线性结构,所以查询某一个节点的时候

  7. 【Java入门】使用Java实现文件夹的遍历 - 2

    遍历文件夹我们通常是使用递归进行操作,这种方式比较简单,也比较容易理解。本文为大家介绍另一种不使用递归的方式,由于没有使用递归,只用到了循环和集合,所以效率更高一些!一、使用递归遍历文件夹整体思路1、使用File封装初始目录,2、打印这个目录3、获取这个目录下所有的子文件和子目录的数组。4、遍历这个数组,取出每个File对象4-1、如果File是否是一个文件,打印4-2、否则就是一个目录,递归调用代码实现publicclassSearchFile{publicstaticvoidmain(String[]args){//初始目录Filedir=newFile("d:/Dev");Datebeg

  8. java - 为什么 ruby​​ modulo 与 java/other lang 不同? - 2

    我基本上来自Java背景并且努力理解Ruby中的模运算。(5%3)(-5%3)(5%-3)(-5%-3)Java中的上述操作产生,2个-22个-2但在Ruby中,相同的表达式会产生21个-1-2.Ruby在逻辑上有多擅长这个?模块操作在Ruby中是如何实现的?如果将同一个操作定义为一个web服务,两个服务如何匹配逻辑。 最佳答案 在Java中,模运算的结果与被除数的符号相同。在Ruby中,它与除数的符号相同。remainder()在Ruby中与被除数的符号相同。您可能还想引用modulooperation.

  9. java - Ruby 相当于 Java 的 Collections.unmodifiableList 和 Collections.unmodifiableMap - 2

    Java的Collections.unmodifiableList和Collections.unmodifiableMap在Ruby标准API中是否有等价物? 最佳答案 使用freeze应用程序接口(interface):Preventsfurthermodificationstoobj.ARuntimeErrorwillberaisedifmodificationisattempted.Thereisnowaytounfreezeafrozenobject.SeealsoObject#frozen?.Thismethodretur

  10. java - Java 的 StringReader 的 Ruby 等价物是什么? - 2

    在Java中,可以像这样从一个字符串创建一个IO流:Readerr=newStringReader("mytext");我希望能够在Ruby中做同样的事情,这样我就可以获取一个字符串并将其视为一个IO流。 最佳答案 r=StringIO.new("mytext")和here'sthedocumentation. 关于java-Java的StringReader的Ruby等价物是什么?,我们在StackOverflow上找到一个类似的问题: https://st

随机推荐