当 Java 多态遇到方法重载和重写

关于 Java 多态 (Polymorphism)的一些 tips。

多态定义

  • 多态是同一个行为具有多个不同表现形式或形态的能力。

  • 多态就是同一个接口,使用不同的实例而执行不同操作,如图所示:

在方法参数里的多态

多态的一般情况

对于多态,我们最熟悉的一种情况就是 引用类似被声明为父类后,可以把子类的实例赋值给这个引用。

Object any = new ArrayList<String>();
Object anyone = new String();
List<String> list = new LinkedList<>();

Object 是所有类型的父类,因此他声明的引用,可以执行任意他的子类对象,List 也是同理。

同时,我们也知道这个时候,这个引用在默认情况下也只能访问父类的属性和方法。比如这里的 any, 虽然想表达的是一个 ArrayList 对象,但是也只能访问 Object 的方法和属性。

当 Java 多态遇到方法重载和重写

当然,这里可以做强制类型转换。

当 Java 多态遇到方法重载和重写

这里只是举例,一般不会这么写代码

方法声明和方法被调用

关于多态,我们还知道,如果父类和子类声明了相同的方法,那么运行时,会执行子类中的方法。

现在,再来考虑下面这种情况。

当 Java 多态遇到方法重载和重写

这里使用截图,没用代码块主要是想体现一下 print 方法的调用情况,可以看到这里第一个 print 方法是高亮的

结果会是什么呢?

output

son invoked
self ====== Son

可以看到,最终调用的是 Son 的 self 方法,但是执行的是第一个 print 方法,也就是参数类型为 Father 的方法。由此可见,方法匹配(暂时这么叫吧)是按声明类型执行的,但是在运行期,是按对象的实际类型执行的。

这里很容易在不经意间产生 bug。假设 print(Father param) 是后添加的,那么他在无形中屏蔽掉子类为参数的方法。如果这两个方法除了调用具体类型的方法,其他逻辑有差异的话,就非常危险了。

总结

以上

  • self() 方法的两次实现,其实是方法的重写;
  • print()方法的两次实现,其实是方法的重载;

因此,便有一下规律

Java的方法分派分为两种:

  • 静态分派 – 方法重载分派

    • 编译器就确定

    • 依据调用者的声明类型和方法的参数类型匹配

  • 动态分派 – 方法重写分派

    • 运行时确定

    • 依据调用者的实际类型分派

原文 

https://rebooters.github.io/2020/03/14/Java-多态/

本站部分文章源于互联网,本着传播知识、有益学习和研究的目的进行的转载,为网友免费提供。如有著作权人或出版方提出异议,本站将立即删除。如果您对文章转载有任何疑问请告之我们,以便我们及时纠正。

PS:推荐一个微信公众号: askHarries 或者qq群:474807195,里面会分享一些资深架构师录制的视频录像:有Spring,MyBatis,Netty源码分析,高并发、高性能、分布式、微服务架构的原理,JVM性能优化这些成为架构师必备的知识体系。还能领取免费的学习资源,目前受益良多

转载请注明原文出处:Harries Blog™ » 当 Java 多态遇到方法重载和重写

赞 (0)
分享到:更多 ()

评论 0

  • 昵称 (必填)
  • 邮箱 (必填)
  • 网址