Java8新特性之默认方法详解

目录
  • 简介
  • 正文
    • 什么是默认方法
    • 为啥要提供默认方法呢?
    • 如果不提供呢?
    • 主要针对谁?
    • 实现了默认方法的接口和抽象类有区别吗
    • 是不是可以说Java现在也实现了多重继承?
  • 总结

简介

在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修饰且包含方法内容的方法
  • 为什么要提供默认方法:向后兼容,使系统平滑过渡;主要针对类库设计者
  • 多重继承带来的问题:二义性,也叫菱形问题;解决办法就是子类尽量覆写默认方法并显式声明调用哪个方法(实际上这个问题很少出现,因为它属于编译错误,写代码时随时可以发现)

到此这篇关于Java8新特性之默认方法的文章就介绍到这了,更多相关Java8默认方法内容请搜索我们以前的文章或继续浏览下面的相关文章希望大家以后多多支持我们!

(0)

相关推荐

  • Java8中的默认方法(面试者必看)

    背景 在Java8之前,定义在接口中的所有方法都需要在接口实现类中提供一个实现,如果接口的提供者需要升级接口,添加新的方法,那么所有的实现类都需要把这个新增的方法实现一遍,如果说所有的实现类能够自己控制的话,那么还能接受,但是现实情况是实现类可能不受自己控制.比如说Java中的集合框架中的List接口添加一个方法,那么Apache Commons这种框架就会很难受,必须修改所有实现了List的实现类 现在的接口有哪些不便 向已经发布的接口中添加新的方法是问题的根源,一旦接口发生变化,接口的实现者

  • 谨慎使用Java8的默认方法

    默认方法给JVM的指令集增加了一个非常不错的新特性.使用了默认方法之后,如果库中的接口增加了新的方法,实现了这个接口的用户类能够自动获得这个方法的默认实现.一旦用户想更新他的实现类的话,只需覆盖一下这个默认方法就可以了,取而代之的是一个在特定场景下更有意义的实现.更棒的是,用户可以在重写的方法里面调用接口的默认实现来增加一些额外的功能. 目前为止一切都还不错.然而,给现有的Java接口增加默认方法可能会导致代码的不兼容.看个例子就很容易能明白了.假设有一个库,它需要用户实现它的一个接口作为输入:

  • Java8新特性之默认方法(default)浅析

    一.什么是默认方法,为什么要有默认方法 简单说,就是接口可以有实现方法,而且不需要实现类去实现其方法.只需在方法名前面加个default关键字即可. 为什么要有这个特性?首先,之前的接口是个双刃剑,好处是面向抽象而不是面向具体编程,缺陷是,当需要修改接口时候,需要修改全部实现该接口的类,目前的java 8之前的集合框架没有foreach方法,通常能想到的解决办法是在JDK里给相关的接口添加新的方法及实现.然而,对于已经发布的版本,是没法在给接口添加新方法的同时不影响已有的实现.所以引进的默认方法

  • 一篇文章带你认识Java8接口的默认方法

    前言 Java8是Oracle于2014年3月发布的一个重要版本,其API在现存的接口上引入了非常多的新方法. 例如,Java8的List接口新增了sort方法.在Java8之前,则每个实现了List接口的类必须定义sort方法的实现,或者从父类中继承它的实现.想象一下,如果List接口的继承体系非常庞杂,那么整个集合框架的维护量有多么大! 为此,在Java8中引入了一种新的机制:接口支持申明带实现的方法. 默认方法 前文提到了Java8中List接口新增了sort方法,其源码如下: publi

  • Java8中新特性Optional、接口中默认方法和静态方法详解

    前言 毫无疑问,Java 8是Java自Java 5(发布于2004年)之后的最重要的版本.这个版本包含语言.编译器.库.工具和JVM等方面的十多个新特性. Java 8是Java的一个重大版本,有人认为,虽然这些新特性领Java开发人员十分期待,但同时也需要花不少精力去学习.下面本文就给大家详细介绍了Java8中新特性Optional.接口中默认方法和静态方法的相关内容,话不多说了,来一起看看详细的介绍吧. Optional Optional 类(java.util.Optional) 是一个

  • 30分钟入门Java8之默认方法和静态接口方法学习

    前言 上一篇文章30分钟入门Java8之lambda表达式,我们学习了lambda表达式.现在继续Java8新语言特性的学习,今天,我们要学习的是默认方法和静态接口方法. 这一Java8的新语言特性,在Android N中也得到了支持.至于如何在Android开发中配置Java8的开发环境,请查看上一篇文章30分钟入门Java8之lambda表达式. 默认方法 默认方法让我们能给我们的软件库的接口增加新的方法,并且能保证对使用这个接口的老版本代码的兼容性. 下面通过一个简单的例子来深入理解下默认

  • Java8新特性之默认方法和静态方法

    前言 在Java 8之前,默认情况下,接口中的所有方法都是公共的和抽象的.但是这一限制在Java 8中被打破了,Java 8允许开发人员在接口中添加新方法,而无需在实现这些接口的类中进行任何更改. 为什么会有默认方法? 主要是为了方便扩展已有接口:如果没有默认方法,假如给Java中的某个接口添加一个新的抽象方法,那么所有实现了该接口的类都得修改,影响将非常大. 举个例子,Sortable <T>接口以及实现该接口的类SortableNumberCollection和SortableString

  • Java8默认方法Default Methods原理及实例详解

    这篇文章主要介绍了Java8默认方法Default Methods原理及实例详解,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下 Java 8 引入了新的语言特性--默认方法(Default Methods). Default methods enable new functionality to be added to the interfaces of libraries and ensure binary compatibility wit

  • java8新特性之接口默认方法示例详解

    前言 JAVA8 已经发布很久,而且毫无疑问,java8 是自 java5(2004年发布)之后的最重要的版本.其中包括语言.编译器.库.工具和 JVM 等诸多方面的新特性.Java8 新特性列表如下: 接口默认方法 函数式接口 Lambda 表达式 方法引用 Stream Optional 类 Date API Base64 重复注解与类型注解 接口默认方法 1.什么是接口默认方法 从 Java8 开始,程序允许在接口中包含带有具体实现的方法,使用 default 修饰,这类方法就是默认方法.

  • Java8接口的默认方法

    Java8接口的默认方法 什么是默认方法,为什么要有默认方法? 简单说,就是接口可以有实现方法,而且不需要实现类去实现其方法.只需在方法名前面加个default关键字即可. 为什么要有这个特性?首先,之前的接口是个双刃剑,好处是面向抽象而不是面向具体编程,缺陷是,当需要修改接口时候,需要修改全部实现该接口的类,目前的 java 8之前的集合框架没有foreach方法,通常能想到的解决办法是在JDK里给相关的接口添加新的方法及实现.然而,对于已经发布的版本,是没法在给接口添加新方法的同时不影响已有

随机推荐