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

 更新时间:2019年05月18日 10:41:55   作者:sum41丶  
这篇文章主要给大家介绍了如何通过一篇文章带你认识Java8接口的默认方法的相关资料,文中通过示例代码介绍的非常详细,对大家学习或者使用Java8具有一定的参考学习价值,需要的朋友们下面来一起学习学习吧

前言

Java8是Oracle于2014年3月发布的一个重要版本,其API在现存的接口上引入了非常多的新方法。

例如,Java8的List接口新增了sort方法。在Java8之前,则每个实现了List接口的类必须定义sort方法的实现,或者从父类中继承它的实现。想象一下,如果List接口的继承体系非常庞杂,那么整个集合框架的维护量有多么大!

为此,在Java8中引入了一种新的机制:接口支持申明带实现的方法。

默认方法

前文提到了Java8中List接口新增了sort方法,其源码如下:

public interface List<E> extends Collection<E> {
 
 // ...其他成员
  
 default void sort(Comparator<? super E> c) {
  ...
  ...
 }
}

可以看到,这个新增的sort方法有方法体,由default修饰符修饰,这就是接口的默认方法。

很显然,默认方法不是static的,所以必须由接口的实现类的实例来调用这些默认方法。

下面自定义一个接口,练习使用默认方法。

public interface Sized {
 // 普通抽象方法,默认是public abstract修饰的,没有方法体
 int size();

 /*
  * 默认方法,有方法体
  * 任何一个实现了Sized接口的类都会向动继承isEmpty的实现
  */
 default boolean isEmpty() {
  return this.size() == 0;
 }
}

其实,随着JDK版本的不断升级,API在不断演进,默认方法在Java8的API中已经大量地使用了,上面List接口中的sort方法就是其中一个。

和抽象类的区别

有同学可能发现了,Java8中加入了默认方法的接口,这不就是以前的抽象类吗?其实,两者还是有区别的。

  • 一个类只能继承一个抽象类;但是一个类可以实现多个接口。
  • 抽象类有实例变量,而接口只能有类变量

解决冲突

我们知道Java语言中一个类只能继承一个父类,但是一个类可以实现多个接口。随着默认方法在Java8中的引入,有可能出现一个类继承了多个签名一样的方法。这种情况下,类会选择使用哪一个函数呢?

为解决这种多继承关系,Java8提供了下面三条规则:

  • 类中的方法优先级最高,类或父类中声明的方法的优先级高于任何声明为默认方法的优先级。
  • 如果第一条无法判断,那么子接口的优先级更高:方法签名相同时,优先选择拥有最具体实现的默认方法的接口, 即如果B继承了A,那么B就比A更加具体。
  • 最后,如果还是无法判断,继承了多个接口的类必须通过显式覆盖和调用期望的方法, 显式地选择使用哪一个默认方法的实现

让我们一起看几个例子 。

场景1:

public interface A {
 default void hello() {
  System.out.println("hello from A");
 }
}
public interface B extends A {
 default void hello() {
  System.out.println("hello from B");
 }
}
public class C implements A, B {
 public static void main(String[] args) {
  new C().hello();
 }
}


如图1,是这个场景的UML图。

我们对照上面三条规则来看,类C中main()方法会输出什么?

  1. 规则(1))不满足。
  2. 因为B继承了A,所以B比A更具体,所以应该选择B的hello()方法。所以,程序会打印输出"hello from B"。

场景2:

如果C像下面这样继承了D,会怎么样?

public class D implements A {

}
public class C extends D implements A, B {
 public static void main(String[] args) {
  new C().hello();
 }
}


如图2,是这个场景的UML图。

同样,我们对照着三条规则来看:

  1. C虽然继承了D,但D中未覆盖A的默认方法。
  2. 接着,编译器会在A和B中做选择,由于B更具体,所以,程序会打印输出"hello from B"。

场景3:

将上面的D稍作修改:

public class D implements A {
 public void hello() {
  System.out.println("hello from D");
 }
}

结果又如何?

由于依据规则(1),父类中声明的方法具有更高的优先级,所以程序会打印输出"hello from D"。

场景4:

假设现在B不在继承A:

public interface A {
  default void hello() {
    System.out.println("hello from A");
  }
}
public interface B {
  default void hello() {
    System.out.println("hello from B");
  }
}
public class C implements A, B {
  public static void main(String[] args) {
    new C().hello();
  }
}

如图3,是这个场景的UML图。

此时,由于编译器无法识别A还是B的实现更加具体,所以会抛出编译错误:”C inherits unrelated defaults for hello() from types A and B“。

像这种场景要解决冲突,可以在C中覆盖hello()方法并在方法内显示的选择调用A还是B的方法。

调用方式如下:

public class C extends D implements A, B {
  public void hello() {
    // 显式地选择调用接口B中的方法
    // 同理,要调用接口A中的方法,可以这样:A.super.hello()
    B.super.hello();
  }

  public static void main(String[] args) {
    // 输出 hello from B
    new C().hello();
  }
}

场景5:

public interface A {
  default void hello() {
    System.out.println("hello from A");
  }
}
public interface B extends A{

}
public interface C extends A{

}
public class D implements B, C {
  public void hello() {
    new D().hello();
  }
}

此时,只有一个方法申明可以选择,所以程序会输出"hello from A"。

总结

以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,谢谢大家对脚本之家的支持。

相关文章

  • Java中static变量作用和用法详解

    Java中static变量作用和用法详解

    Java 中被 static 修饰的成员称为静态成员或类成员。它属于整个类所有,而不是某个对象所有,即被类的所有对象所共享。静态成员可以使用类名直接访问,也可以使用对象名进行访问.下面我们来详细了解一下吧
    2019-06-06
  • springboot+redis过期事件监听实现过程解析

    springboot+redis过期事件监听实现过程解析

    这篇文章主要介绍了springboot+redis过期事件监听实现过程解析,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下
    2020-03-03
  • IntelliJ IDEA弹出“IntelliJ IDEA License Activation”的处理方法

    IntelliJ IDEA弹出“IntelliJ IDEA License Activation”的处理方法

    这篇文章主要介绍了IntelliJ IDEA弹出“IntelliJ IDEA License Activation”的处理方法,本文给出解决方法,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2020-09-09
  • Hibernate的一对一,一对多/多对一关联保存的实现

    Hibernate的一对一,一对多/多对一关联保存的实现

    本文主要介绍了Hibernate的一对一,一对多/多对一关联保存的实现,文中通过示例代码介绍的很详细,感兴趣的可以了解一下
    2021-09-09
  • Mybatis-plus数据权限DataPermissionInterceptor实现

    Mybatis-plus数据权限DataPermissionInterceptor实现

    本文主要介绍了Mybatis-plus数据权限DataPermissionInterceptor实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2022-07-07
  • Java中随机函数变换的示例详解

    Java中随机函数变换的示例详解

    这篇文章主要为大家详细介绍了Java中随机函数的变换,文中的示例代码讲解详细,对我们学习Java有一定的帮助,感兴趣的可以了解一下
    2022-08-08
  • springboot如何获取文件流

    springboot如何获取文件流

    这篇文章主要介绍了springboot如何获取文件流,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2022-03-03
  • Java实现一个简单计算器

    Java实现一个简单计算器

    这篇文章主要介绍了Java实现一个简单计算器,文章我围绕实现简单计算器的相关代码展现全文,具有一定的参考价值,需要的小伙伴可以参考一下,
    2022-01-01
  • zookeeper的Leader选举机制源码解析

    zookeeper的Leader选举机制源码解析

    这篇文章主要为大家介绍了zookeeper的Leader选举源码解析,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2023-03-03
  • javaweb实现文件上传功能

    javaweb实现文件上传功能

    这篇文章主要为大家详细介绍了javaweb实现文件上传功能,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2022-06-06

最新评论