Java8中的默认方法
作者:汤圆
个人博客:javalover.cc
前言
大家好啊,我是汤圆,今天给大家带来的是《Java8中的默认方法》,希望对大家有帮助,谢谢
文章纯属原创,个人总结难免有差错,如果有,麻烦在评论区回复或后台私信,谢啦
简介
在Java之前,我们接触到的接口,都是只定义方法,不实现方法
(你看下面这几个人,像不像接口)
但是到了Java8就不一样了,因为在接口中新增了默认方法
这样的话,有些活,就可以交给接口自己去做了,而不用实现类去做(Java你这是在收买人心啊)
我们下面以问答的形式来介绍默认方法的相关知识点(据说问答模式可以让人更好地记忆?)
目录
什么是默认方法?
为啥要提供默认实现?
那我如果不提供呢?
这个功能主要是针对谁?
实现了默认方法的接口和抽象类有区别吗?
是不是可以说实现了多重继承?
正文
什么是默认方法
默认方法是接口中用default修饰的方法,其中包含方法内容
比如下面这个:
public interface InterfaceDemo {
// 普通方法,只定义,不实现
void oldFun();
// 默认方法,又定义,又实现
default void newFun(){
System.out.println("newFun");
}
}
为啥要提供默认方法呢?
为了向后兼容(这也是导致Java变得臃肿的原因之一)。
因为升级系统时,难免会有一些新功能需要加入,此时如果接口类新增了方法,那么实现类就必须同步修改实现;
这样工作量还是很大的,而且很容易出错。
所以Java8开始,推出了接口的默认方法这个功能,使得接口升级变得更加平滑
比如下面的代码:InterfaceDemo就是上面那个接口
public class UserDemo implements InterfaceDemo{
@Override
public void oldFun() {
System.out.println("oldFun");
}
public static void main(String[] args) {
UserDemo demo = new UserDemo();
/**
* InterfaceDemo升级后,新增了newFun方法
* 但是由于newFun是默认方法,有提供实现内容
* 所以这里的子类 UserDemo就可以直接使用
*/
demo.newFun();
}
}
我们可以看到,UserDemo没有实现新的方法newFun(),但是也可以编译运行,并直接调用newFun()
这就是默认方法的好处:对实现类来说是无痛升级的
如果不提供呢?
不提供的话,接口类升级时,系统有两个选择
- 实现类升级:
- 实现类老老实实地按照接口升级后的方法,进行同步修改实现,但是工作量大
- 实现类不升级:
- 实现类不升级也是可以的,只要不引入接口类的新版本就可以了,那么这个时候系统还是可以运行的,这没啥问题。但是谁能保证一辈子都不更新系统呢?如果更新系统时,接口类库升级到新版本,那么编译还是通不过
主要针对谁?
接口的默认方法主要是针对类库设计者
实现了默认方法的接口和抽象类有区别吗
区别没有之前那么多,但还是有的:
- 抽象类单继承,接口类多实现
- 抽象类中的属性定义时不需要初始化,接口类的属性定义时要初始化(默认修饰符为public static final)
是不是可以说Java现在也实现了多重继承?
可以这么说。
但是现在面临的一个新问题,就是多重继承带来的二义性问题,有点类似之前介绍的致命方块(也叫菱形问题)
如下面的UML图所示
比如上面这种,你无法知道A会调用哪个接口的fun方法
所以编译器会报错:
com.jalon.java8.defaultmethod.A inherits unrelated defaults for fun() from types com.jalon.java8.defaultmethod.B and com.jalon.java8.defaultmethod.C
解决办法:
- 先覆写fun方法
- 再显示声明调用哪个接口的fun方法
代码如下:
public class A implements B,C{
@Override
public void fun(){
// 显示调用B的默认方法
B.super.fun();
}
public static void main(String[] args) {
A a = new A();
// 这里会打印B的fun
a.fun();
}
}
interface D{
default void fun(){
System.out.println("D");
}
}
interface B extends D{
@Override
default void fun(){
System.out.println("B");
}
}
interface C extends D{
@Override
default void fun(){
System.out.println("C");
}
}
总结
什么是默认方法:接口中用default修饰且包含方法内容的方法
为什么要提供默认方法:向后兼容,使系统平滑过渡;主要针对类库设计者
多重继承带来的问题:二义性,也叫菱形问题;解决办法就是子类尽量覆写默认方法并显式声明调用哪个方法(实际上这个问题很少出现,因为它属于编译错误,写代码时随时可以发现)
后记
最后,感谢大家的观看,谢谢
微信搜【汤圆学Java】:
- 回复【Java】免费领取 Java核心书籍-电子版
- 回复【Idea】免费领取 Idea激活资源
最新文章
- ASP.NET内置对象的总结
- xmpp关于后台挂起的消息接收,后台消息推送,本地发送通知
- 创建多个Oracle数据库及相应的实例
- Python学习笔记- Python threading模块
- 自定义View--一个简单地圆形Progress效果
- Delphi逆向
- 批量产生ssh2项目中hibernate带注解的pojo类的快捷方法
- Android开发书籍推荐
- JSP三大常用对象request、response、session
- OpenCV面、人眼检测
- Java程序员的10道XML面试题 (转)
- IIS配置發佈網站常見問題及設置
- Niagara 泵阀
- Python实战之logging模块使用详解
- NOIp模拟赛 巨神兵(状压DP 容斥)
- 解决网页响应慢,waiting(TTFB)时间过长
- 20145118《Java程序设计》 第8周学习总结
- 310实验室OTL问题(2)
- python 浮点数转分数
- LR之Java Vuser II
热门文章
- Chrome 80 &; SameSite &; cookie
- Flutter: Draggable和DragTarget 可拖动小部件
- 人物传记Kyle Tedford:数据环境生变,银行大数据风控怎么办?
- mysql一张表到底能存多少数据?
- 利用 Java 操作 Jenkins API 实现对 Jenkins 的控制详解
- 2021 年学习 React 的所需要的 JavaScript 基础
- 详细探秘Linux 和 Window 双系统访问Windows 磁盘需要输入密码问题解决过程分析
- IVMS-5000海康平台安装
- linux调度全景指南
- 后端程序员之路 16、信息熵 、决策树、ID3