Java 8新特性——default方法(defender方法)介绍

我们都知道在Java语言的接口中只能定义方法名,而不能包含方法的具体实现代码。接口中定义的方法必须在接口的非抽象子类中实现。下面就是关于接口的一个例子:


1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

public interface SimpleInterface {

  public void doSomeWork();

}

class SimpleInterfaceImpl implements SimpleInterface{

  @Override

  public void doSomeWork() {

    System.out.println("Do Some Work implementation in the class");

  }

  public static void main(String[] args) {

    SimpleInterfaceImpl simpObj = new SimpleInterfaceImpl();

    simpObj.doSomeWork();

  }

}

那么,如果我们在SimpleInterface里面添加一个新方法,会怎样呢?


1

2

3

4

public interface SimpleInterface {

  public void doSomeWork();

  public void doSomeOtherWork();

}

如果我们尝试编译上面的这段代码,会得到如下结果:


1

2

3

4

5

6

$javac .\SimpleInterface.java

.\SimpleInterface.java:18: error: SimpleInterfaceImpl is not abstract and does not

override abstract method doSomeOtherWork() in SimpleInterface

class SimpleInterfaceImpl implements SimpleInterface{

^

1 error

因为接口有这个语法限制,所以要直接改变/扩展接口内的方法变得非常困难。我们在尝试强化Java 8 Collections API,让其支持lambda表达式的时候,就面临了这样的挑战。为了克服这个困难,Java 8中引入了一个新的概念,叫做default方法,也可以称为Defender方法,或者虚拟扩展方法(Virtual extension methods)。

Default方法是指,在接口内部包含了一些默认的方法实现(也就是接口中可以包含方法体,这打破了Java之前版本对接口的语法限制),从而使得接口在进行扩展的时候,不会破坏与接口相关的实现类代码。接下来,让我们看一个例子:


1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

public interface SimpleInterface {

  public void doSomeWork();

  //A default method in the interface created using "default" keyword

  //使用default关键字创在interface中直接创建一个default方法,该方法包含了具体的实现代码

  default public void doSomeOtherWork(){

    System.out.println("DoSomeOtherWork implementation in the interface");

  }

}

class SimpleInterfaceImpl implements SimpleInterface{

  @Override

  public void doSomeWork() {

    System.out.println("Do Some Work implementation in the class");

  }

  /*

   * Not required to override to provide an implementation

   * for doSomeOtherWork.

   * 在SimpleInterfaceImpl里,不需要再去实现接口中定义的doSomeOtherWork方法

   */

  public static void main(String[] args) {

    SimpleInterfaceImpl simpObj = new SimpleInterfaceImpl();

    simpObj.doSomeWork();

    simpObj.doSomeOtherWork();

  }

}

该程序的输出是:


1

2

Do Some Work implementation in the class

DoSomeOtherWork implementation in the interface

以上是对default方法的一个非常简要的介绍。如果有兴趣的话,还可以通过看这篇文档,来获取更深层次的理解。

更新:

现在大家问得比较多的一个问题是:如果一个类实现了两个接口(可以看做是“多继承”),这两个接口又同时都包含了一个名字相同的default方法,那么会发生什么情况? 在这样的情况下,编译器会报错。让我用例子来解释一下:


1

2

3

4

5

6

7

8

9

10

11

public interface InterfaceWithDefaultMethod {

  public void someMethod();

  default public void someOtherMethod(){

    System.out.println("Default method implementation in the interface");

  }

}

public interface InterfaceWithAnotherDefMethod {

  default public void someOtherMethod(){

    System.out.println("Default method implementation in the interface");

  }

}

然后我们定义一个类,同时实现以上两个接口:


1

2

3

4

5

6

7

8

9

10

11

12

13

public class DefaultMethodSample implements

  InterfaceWithDefaultMethod, InterfaceWithAnotherDefMethod{

  @Override

  public void someMethod(){

    System.out.println("Some method implementation in the class");

  }

  public static void main(String[] args) {

    DefaultMethodSample def1 = new DefaultMethodSample();

    def1.someMethod();

    def1.someOtherMethod();

  

}

如果编译以上的代码,会得到一个编译器错误,如下所示。因为编译器不知道应该在两个同名的default方法中选择哪一个,因此产生了二义性。

原文链接: blog.sanaulla.info 翻译: ImportNew.com黄小非
译文链接: http://www.importnew.com/7302.html
转载请保留原文出处、译者和译文链接。]

时间: 2024-12-15 07:13:33

Java 8新特性——default方法(defender方法)介绍的相关文章

Java 8新特性-4 方法引用

对于引用来说我们一般都是用在对象,而对象引用的特点是:不同的引用对象可以操作同一块内容! Java 8的方法引用定义了四种格式: 引用静态方法     ClassName :: staticMethodName 引用对象方法:  Object:: methodName 引用特定类型方法: ClassName :: methodName 引用构造方法: ClassName  :: new 静态方法引用示例 /** * 静态方法引用 * @param <P> 引用方法的参数类型 * @param

Java 8新特性

现在,是时候把所有Java8的重要特性收集整理成一篇单独的文章了,希望这篇文章能给你带来阅读上的乐趣.开始吧! 目录结构 介绍 Java语言的新特性 2.1 Lambdas表达式与Functional接口 2.2 接口的默认与静态方法 2.3 方法引用 2.4 重复注解 2.5 更好的类型推测机制 2.6 扩展注解的支持 Java编译器的新特性 3.1 参数名字 Java 类库的新特性 4.1 Optional 4.2 Streams 4.3 Date/Time API (JSR 310) 4.

Java 8 新特性1-函数式接口

Java 8 新特性1-函数式接口 (原) Lambda表达式基本结构: (param1,param2,param3) -> {代码块} 例1: package com.demo.jdk8; import java.util.Arrays; import java.util.List; import java.util.function.Consumer; public class Test2 { public static void main(String[] args) { for_test

Java 8新特性终极指南

目录结构 介绍 Java语言的新特性 2.1 Lambdas表达式与Functional接口 2.2 接口的默认与静态方法 2.3 方法引用 2.4 重复注解 2.5 更好的类型推测机制 2.6 扩展注解的支持 Java编译器的新特性 3.1 参数名字 Java 类库的新特性 4.1 Optional 4.2 Streams 4.3 Date/Time API (JSR 310) 4.4 JavaScript引擎Nashorn 4.5 Base64 4.6 并行(parallel)数组 4.7

Java 8新特性前瞻

快端午小长假了,要上线的项目差不多完结了,终于有时间可以坐下来写篇博客了. 这是篇对我看到的java 8新特性的一些总结,也是自己学习过程的总结. 几乎可以说java 8是目前为止,自2004年java 5发布以来的java世界中最大的事件了.它带来了java语言层面上的诸多改变,主要包括下面一些方面:语法.编译器.库.工具和运行时. 一,语法层面: 1,Lambda表达式. lambda表达式是一种可调用对象,它允许我们将函数作为函数参数传入.诸如C++.Groovy.Scala都已经支持la

Java 8 新特性 – 终极手册整理

1.简介 毫无疑问,Java 8是自Java  5(2004年)发布以来Java语言最大的一次版本升级,Java 8带来了很多的新特性,比如编译器.类库.开发工具和JVM(Java虚拟机).在这篇教程中我们将会学习这些新特性,并通过真实例子演示说明它们适用的场景. 本教程由下面几部分组成,它们分别涉及到Java平台某一特定方面的内容: 语言 编译器 类库 开发工具 运行时(Java虚拟机) 2.Java的新特性 总体来说,Java 8是一个大的版本升级.有人可能会说,Java 8的新特性非常令人

【整理】Java 8新特性总结

闲语: 相比于今年三月份才发布的Java 10 ,发布已久的Java 8 已经算是老版本了(传闻Java 11将于9月25日发布....).然而很多报道表明:Java 9 和JJava10不是 LTS 版本,和过去的 Java 大版本升级不同,它们只有半年左右的开发和维护期.而未来的 Java11,也就是 18.9 LTS,才是 Java 8 之后第一个 LTS 版本(得到 Oracle 等商业公司的长期支持服务).所以Java 8 就成了最新的一次LTS版本升级,这也是为什么Java开发者对J

Java 8新特性之旅:使用Stream API处理集合

在这篇“Java 8新特性教程”系列文章中,我们会深入解释,并通过代码来展示,如何通过流来遍历集合,如何从集合和数组来创建流,以及怎么聚合流的值. 在之前的文章“遍历.过滤.处理集合及使用Lambda表达式增强方法”中,我已经深入解释并演示了通过lambda表达式和方法引用来遍历集合,使用predicate接口来过滤集合,实现接口的默认方法,最后还演示了接口静态方法的实现. 源代码都在我的Github上:可以从 这里克隆. 内容列表 使用流来遍历集合. 从集合或数组创建流. 聚合流中的值. 1.

Spring 4支持的Java 8新特性一览

有众多新特性和函数库的Java 8发布之后,Spring 4.x已经支持其中的大部分.有些Java 8的新特性对Spring无影响,可以直接使用,但另有些新特性需要Spring的支持.本文将带您浏览Spring 4.0和4.1已经支持的Java 8新特性. Spring 4支持Java 6.7和8 Java 8编译器编译过的代码生成的.class文件需要在Java 8或以上的Java虚拟机上运行.由于Spring对反射机制和ASM.CGLIB等字节码操作函数库的重度使用,必须确保这些函数库能理解