作者:汤圆

个人博客: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()

这就是默认方法的好处:对实现类来说是无痛升级的

如果不提供呢?

不提供的话,接口类升级时,系统有两个选择

  1. 实现类升级:

    • 实现类老老实实地按照接口升级后的方法,进行同步修改实现,但是工作量大
  2. 实现类不升级:
    • 实现类不升级也是可以的,只要不引入接口类的新版本就可以了,那么这个时候系统还是可以运行的,这没啥问题。但是谁能保证一辈子都不更新系统呢?如果更新系统时,接口类库升级到新版本,那么编译还是通不过

主要针对谁?

接口的默认方法主要是针对类库设计者

实现了默认方法的接口和抽象类有区别吗

区别没有之前那么多,但还是有的:

  1. 抽象类单继承,接口类多实现
  2. 抽象类中的属性定义时不需要初始化,接口类的属性定义时要初始化(默认修饰符为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激活资源

Java8中的默认方法的更多相关文章

  1. 【Java8新特性】接口中的默认方法和静态方法,你都掌握了吗?

    写在前面 在Java8之前的版本中,接口中只能声明常量和抽象方法,接口的实现类中必须实现接口中所有的抽象方法.而在Java8中,接口中可以声明默认方法和静态方法,本文,我们就一起探讨下接口中的默认方法 ...

  2. Java8接口中的默认方法

    Java8新增特性,可以为接口中添加默认方法,实现这个接口的所有类都会继承这个方法,这样看起来,接口和类的界限就有点不明显了,同时也会带来多继承,菱形问题.这样设计的初衷是什么? 重所周知,java8 ...

  3. Java8新特性Optional、接口中的默认方法与静态方法

    Optional Optional 类(java.util.Optional) 是一个容器类,代表一个值存在或不存在,原来用 null 表示一个值不存在,现在 Optional 可以更好的表达这个概念 ...

  4. JDK8.0接口中的默认方法和静态方法

    我们在接口中通常定义的方法是抽象方法,即没有方法体,只有返回值类型和方法名:(public abstract) void Method(); 类在实现接口的时候必须重写抽象方法才可以 jdk8中新加的 ...

  5. Java 8 接口中的默认方法与静态方法

    Java 8 接口中的默认方法与静态方法 1. 接口中的默认方法 允许接口中包含具有具体实现的方法,该方法称"默认方法",默认方法使用用 default 关键字修饰. public ...

  6. Java8新特性_接口中的默认方法

    默认方法由来猜想 1. Collection接口.Collections公共类.  同是操作集合,为啥要搞俩?没必要.在接口中搞一些默认实现,一个接口即搞定了. 2. Java8支持Lambda表达式 ...

  7. java8新特性-默认方法

    作为一个java程序猿,经常会被问基础怎么样,对于这个问题,我理解的有两方面:一是对于java基础的理解和掌握,比如JDK的相关特性:二是工作的经历,毕竟,语言编程是一门实战性质的艺术,就算掌握了千万 ...

  8. java8新特性- 默认方法 在接口中有具体的实现

    案例分析 在java8中在对list循环的时候,我们可以使用forEach这个方法对list进行遍历,具体代码如下demo所示 public static void main(String[] arg ...

  9. Java8 新特性 默认方法

    默认方法为什么出现 默认方法的出现是因为在java8设计的过程中,因为加入了Lamdba表达式,和函数式接口,所以在非常多的接口里面要加入新的方法,但是如果在接口里面直接加入新的方法,那么以前写的所有 ...

随机推荐

  1. TypeScript & Advanced Types

    TypeScript & Advanced Types https://www.typescriptlang.org/docs/handbook/advanced-types.html#typ ...

  2. moment.js 时间格式转换

    moment.js 时间格式转换 moment.js 时间转化 bug 格式错误 bug 02:00 => 14:00 format HH 与 hh HH === 24 小时制 hh === 1 ...

  3. Golang 实现 Redis(9): 使用GeoHash 搜索附近的人

    本文是使用 golang 实现 redis 系列的第九篇,主要介绍如何使用 GeoHash 实现搜索附近的人. 搜索附近的POI是一个非常常见的功能,它的技术难点在于地理位置是二维的(经纬度)而我们常 ...

  4. 构建Docker私有仓库

    一.Docker私有仓库   上一篇说了如何利用Dockerfile在已有镜像的基础上构建自己的镜像,那么如果需要让镜像在一个团队中使用,就需要一个仓库,有几种方式可以共享私有镜像. 1.将镜像上传至 ...

  5. 【资源下载】安卓VS鸿蒙第三方件切换宝典 V1.0

    下载<安卓VS鸿蒙第三方件切换宝典> 由于字数较多,本文仅展示部分,查看完整版请点击上方下载 众所周知,安卓应用开发经过这么多年的发展相对成熟和稳定,鸿蒙OS作为后来者兼容一个成熟的开发体 ...

  6. python的基本运算符

    目录 基本运算符 1.算术运算符 2.比较运算符 3.赋值运算符 4.逻辑运算符 5.身份运算符 6.位运算符 7.成员运算符 基本运算符 1.算术运算符 运算符 描述 实例 + 加-两个对象相加 a ...

  7. lambda表达式在python和c++中的异同

    Lambda表达式是干么的?.lambda表达式首先是一个表达式,是一个函数对象一个匿名函数,但不是函数.现在流行语言例如:JS.PHP都支持一种和面向过程.面向对象并列的函数式编程,lambda就是 ...

  8. AOP面试造火箭始末

    本文已整理致我的github地址,欢迎大家 star 支持一下 这是一个困扰我司由来已久的难题,Dubbo 了解过吧,对外提供的服务可能有多个方法,一般我们为了不给调用方埋坑,会在每个方法里把所有异常 ...

  9. Vue入门干货,以及遇到的坑

    一.安装环境及Vue脚手架搭建 参考文档:https://www.jianshu.com/p/1626b8643676 二.开发文档 官方文档:https://cn.vuejs.org/v2/guid ...

  10. 奇思妙想 CSS 文字动画

    之前有些过两篇关于字体的文章,是关于如何定义字体的: 你该知道的字体 font-family Web 字体 font-family 再探秘 本文将会和这篇 -- CSS 奇思妙想边框动画类似,讲一些文 ...