百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 文章教程 > 正文

讲解一下Java 中的多态

xsobi 2025-01-03 19:37 1 浏览

多态(Polymorphism)属于面向对象三大特征之一,它的前提是封装形成独立体,独立体之间存在继承关系,从而产生多态机制。多态是同一个行为具有多个不同表现形式或形态的能力。

重载式多态,也叫编译时多态。也就是说这种多态再编译时已经确定好了。重载大家都知道,方法名相同而参数列表不同的一组方法就是重载。在调用这种重载的方法时,通过传入不同的参数最后得到不同的结果。

但是这里是有歧义的,有的人觉得不应该把重载也算作多态。因为很多人对多态的理解是:程序中定义的引用变量所指向的具体类型和通过该引用变量发出的方法调用在编程时并不确定,而是在程序运行期间才确定,这种情况叫做多态。 这个定义中描述的就是我们的第二种多态—重写式多态。并且,重载式多态并不是面向对象编程特有的,而多态却是面向对象三大特性之一(如果我说的不对,记得告诉我。。)。

我觉得大家也没有必要在定义上去深究这些,我的理解是:同一个行为具有多个不同表现形式或形态的能力就是多态,所以我认为重载也是一种多态,如果你不同意这种观点,我也接受。

重写式多态,也叫运行时多态。这种多态通过动态绑定(dynamic binding)技术来实现,是指在执行期间判断所引用对象的实际类型,根据其实际的类型调用其相应的方法。也就是说,只有程序运行起来,你才知道调用的是哪个子类的方法。 这种多态通过函数的重写以及向上转型来实现,我们上面代码中的例子就是一个完整的重写式多态。我们接下来讲的所有多态都是重写式多态,因为它才是面向对象编程中真正的多态。

向上转型

子类引用的对象转换为父类类型称为向上转型。通俗地说就是是将子类对象转为父类对象。此处父类对象可以是接口。

看一个大家都知道的例子:

实例

public class Animal {

public void eat(){

System.out.println("animal eatting...");

}

}


public class Cat extends Animal{


public void eat(){


System.out.println("我吃鱼");

}

}


public class Dog extends Animal{


public void eat(){


System.out.println("我吃骨头");

}


public void run(){

System.out.println("我会跑");

}

}


public class Main {


public static void main(String[] args) {


Animal animal = new Cat(); //向上转型

animal.eat();


animal = new Dog();

animal.eat();

}


}


//结果:

//我吃鱼

//我吃骨头

这就是向上转型,Animal animal = new Cat(); 将子类对象 Cat 转化为父类对象 Animal。这个时候 animal 这个引用调用的方法是子类方法。

转型过程中需要注意的问题

向上转型时,子类单独定义的方法会丢失。比如上面Dog类中定义的run方法,当animal引用指向Dog类实例时是访问不到run方法的,animal.run()会报错。

子类引用不能指向父类对象。Cat c = (Cat)new Animal()这样是不行的。

向上转型的好处

减少重复代码,使代码变得简洁。

提高系统扩展性。

向下转型

与向上转型相对应的就是向下转型了。向下转型是把父类对象转为子类对象。(请注意!这里是有坑的。)

案例驱动

先看一个例子:

//还是上面的animal和cat dog

Animal a = new Cat();

Cat c = ((Cat) a);

c.eat();

//输出 我吃鱼

Dog d = ((Dog) a);

d.eat();

// 报错 : java.lang.ClassCastException:com.chengfan.animal.Cat cannot be cast to com.chengfan.animal.Dog

Animal a1 = new Animal();

Cat c1 = ((Cat) a1);

c1.eat();

// 报错 : java.lang.ClassCastException:com.chengfan.animal.Animal cannot be cast to com.chengfan.animal.Cat

为什么第一段代码不报错呢?相比你也知道了,因为 a 本身就是 Cat 对象,所以它理所当然的可以向下转型为 Cat,也理所当然的不能转为 Dog,你见过一条狗突然就变成一只猫这种操蛋现象?

而 a1 为 Animal 对象,它也不能被向下转型为任何子类对象。比如你去考古,发现了一个新生物,知道它是一种动物,但是你不能直接说,啊,它是猫,或者说它是狗。

向下转型注意事项

向下转型的前提是父类对象指向的是子类对象(也就是说,在向下转型之前,它得先向上转型)

向下转型只能转型为本类对象(猫是不能变成狗的)。

看一个经典案例:

实例

class A {

public String show(D obj) {

return ("A and D");

}


public String show(A obj) {

return ("A and A");

}


}


class B extends A{

public String show(B obj){

return ("B and B");

}


public String show(A obj){

return ("B and A");

}

}


class C extends B{


}


class D extends B{


}


public class Demo {

public static void main(String[] args) {

A a1 = new A();

A a2 = new B();

B b = new B();

C c = new C();

D d = new D();


System.out.println("1--" + a1.show(b));

System.out.println("2--" + a1.show(c));

System.out.println("3--" + a1.show(d));

System.out.println("4--" + a2.show(b));

System.out.println("5--" + a2.show(c));

System.out.println("6--" + a2.show(d));

System.out.println("7--" + b.show(b));

System.out.println("8--" + b.show(c));

System.out.println("9--" + b.show(d));

}

}

//结果:

//1--A and A

//2--A and A

//3--A and D

//4--B and A

//5--B and A

//6--A and D

//7--B and B

//8--B and B

//9--A and D


//能看懂这个结果么?先自分析一下。

前三个,强行分析,还能看得懂。但是第四个,大概你就傻了吧。为什么不是b and b呢?

这里就要学点新东西了。

当父类对象引用变量引用子类对象时,被引用对象的类型决定了调用谁的成员方法,引用变量类型决定可调用的方法。如果子类中没有覆盖该方法,那么会去父类中寻找。

可能读起来比较拗口,我们先来看一个简单的例子:

实例

class X {

public void show(Y y){

System.out.println("x and y");

}


public void show(){

System.out.println("only x");

}

}


class Y extends X {

public void show(Y y){

System.out.println("y and y");

}

public void show(int i){


}

}


class main{

public static void main(String[] args) {

X x = new Y();

x.show(new Y());

x.show();

}

}

//结果

//y and y

//only x

Y 继承了 X,覆盖了 X 中的 show(Y y) 方法,但是没有覆盖 show() 方法。

这个时候,引用类型为X的 x 指向的对象为 Y,这个时候,调用的方法由 Y 决定,会先从 Y 中寻找。执行 x.show(new Y());,该方法在 Y 中定义了,所以执行的是 Y 里面的方法;

但是执行 x.show(); 的时候,有的人会说,Y 中没有这个方法啊?它好像是去父类中找该方法了,因为调用了 X 中的方法。

事实上,Y 类中是有 show() 方法的,这个方法继承自 X,只不过没有覆盖该方法,所以没有在 Y 中明确写出来而已,看起来像是调用了 X 中的方法,实际上调用的还是 Y 中的。

这个时候再看上面那句难理解的话就不难理解了吧。X是引用变量类型,它决定哪些方法可以调用;show()和 show(Y y) 可以调用,而 show(int i)不可以调用。Y 是被引用对象的类型,它决定了调用谁的方法:调用 y 的方法。

上面的是一个简单的知识,它还不足以让我们理解那个复杂的例子。我们再来看这样一个知识:

继承链中对象方法的调用的优先级:this.show(O)、super.show(O)、this.show((super)O)、super.show((super)O)。

如果你能理解这个调用关系,那么多态你就掌握了。我们回到那个复杂的例子:

abcd 的关系是这样的:C/D —> B —> A

我们先来分析4 : a2.show(b)

首先,a2是类型为A的引用类型,它指向类型为B的对象。A确定可调用的方法:show(D obj)和show(A obj)。

a2.show(b) ==> this.show(b),这里this指的是B。然后.在B类中找show(B obj),找到了,可惜没用,因为show(B obj)方法不在可调用范围内,this.show(O)失败,进入下一级别:super.show(O),super指的是A。在A 中寻找show(B obj),失败,因为没用定义这个方法。进入第三级别:this.show((super)O),this指的是B。在B中找show((A)O),找到了:show(A obj),选择调用该方法。

输出:B and A

如果你能看懂这个过程,并且能分析出其他的情况,那你就真的掌握了。

我们再来看一下9:b.show(d)

首先,b为类型为B的引用对象,指向类型为B的对象。没有涉及向上转型,只会调用本类中的方法。

在B中寻找show(D obj),方法。现在你不会说没找到了吧?找到了,直接调用该方法。

输出 A and D。

总结

本篇文章的内容大体上就是这些了。我们来总结一下。

多态,简而言之就是同一个行为具有多个不同表现形式或形态的能力。

多态的分类:运行时多态和编译时多态。

运行时多态的前提:继承(实现),重写,向上转型

向上转型与向下转型。

继承链中对象方法的调用的优先级:this.show(O)、super.show(O)、this.show((super)O)、super.show((super)O)。

相关推荐

Asp.Net快速开发平台(敏捷开发框架)

前言:敏捷开发框架的名称由来呢?我希望开发项目可以结构化的,轻量级的,就像敏捷开发团队一样的高效快速,通过它可以快速开发一个项目。1:什么是敏捷开发框架?答:敏捷开发框架是一款Asp.Net轻量级智能...

Gradio.NET:简化.NET Web应用开发的新利器

...

.NET 8 实现通用权限开发框架

...

干货来了!推荐10个用于C#.NET开发的基本调试工具

今天给各位网友分享10个用于C#.NET开发的基本调试工具,掌握了这10个工具,大家就可以轻松玩转C#.NET开发与调试。话不多说,直接上干活!!1、VisualStudio...

「Net Core开发」webapi 开发

之前的文章:【NetCore开发】C#开发跨平台程序...

.NET5.0和Quartz.NET开发的极简任务调度平台

任务调度是让系统自动化完成特定的任务,在预约的时间点执行任务的过程。任务调度在不同业务需求情况也不一样,有些可能是有着上千上万个任务,需要统一管理;有些可能是为了方便异常进行提醒、监控。项目简介...

3个基于.Net开发的、开源远程管理工具

我是编程乐趣,一个10年.Net开发经验老程序员,点击右上方“关注”,每天为你分享开源项目和编程知识。盘点3个基于.Net开发的、开源Windows远程管理工具。...

.NET 9 中的新增功能:每个开发人员都应该知道的基本更新

...

跨平台开发的未来:如何在 .NET 6 上构建高效的跨平台应用

跨平台开发是当今软件开发领域的一大热点。开发者一直在寻找能够让他们编写一次代码并部署到多个操作系统(如Windows、Linux和macOS)上的工具和框架。传统的开发方式需要针对每个操作系统编...

.Net开发框架最终版将与Win10同步问世

对许多用户而言,7月29日最受关注的事件是Windows10的发布。但事实上,除了Windows10外这一日微软还会带来更多的亮点。届时微软还将发布针对Windows通用应用程序平台的(UWP)....

微软推出 .NET Aspire云端框架:可改进分布式应用开发流程

IT之家5月24日消息,据微软官方新闻稿,微软近日推出一款能够简化.NET云端开发的.NETAspire开发框架,该框架主要包含工具、模板和NuGet包,允许用户“更容易地”创建分布...

VOL.NET6开发MES系统第二篇——基础数据

大家好,我是李工。系统框架我们已经运行起来了,今天我们正式开始搭建MES系统。MES系统主要实现四大模块:基础数据、生产管理、生产数据、操作记录;今天我们主要演示一个单表的增删改查以及权限控制,我们不...

.NET混合开发解决方案1 WebView2简介

在我的博客《...

VOL.NET6开发MES系统第一篇——搭建程序

大家好,我是李工,接下来我将用VOL.NET6搭建一套MES系统;有兴趣的小伙伴们可以一起学习。一、下载源码VOL...

盘点5个基于SkiaSharp开发的.Net开源图形项目

...