Java 8的default方法能作什么?不能作什么?

什么是default方法?

Java 8 发布之后,能够给接口添加新方法,可是,接口仍然能够和它的实现类保持兼容。这很是重要,由于你开发的类库可能正在被多个开发者普遍的使用着。而Java 8以前,在类库中发布了一个接口之后,若是在接口中添加一个新方法,那些实现了这个接口的应用使用新版本的接口就会有崩溃的危险。html

有了Java 8,是否是就没有这种危险了?答案是否认的。java

给接口添加default方法可能会让某些实现类不可用。编程

首先,让咱们看下default方法的细节。bash

在Java 8中,接口中的方法能够被实现(Java8中的static的方法也能够在接口中实现,但这是另外一个话题)。接口中被实现的方法叫作default方法,用关键字default做为修饰符来标识。当一个类实现一个接口的时候,它能够实现已经在接口中被实现过的方法,但这不是必须的。这个类会继承default方法。这就是为何当接口发生改变的时候,实现类不须要作改动的缘由。ide

多继承的时候呢?

当一个类实现了多于一个(好比两个)接口,而这些接口又有一样的default方法的时候,事情就变得很复杂了。类继承的是哪个default方法呢?哪个也不是!在这种状况下,类要本身(直接或者是继承树上更上层的类)来实现default方法(才能够)。学习

当一个接口实现了default方法,另外一个接口把default方法声明成了abstract的时候,一样如此。Java 8试图避免不明确的东西,保持严谨。若是一个方法在多个接口中都有声明,那么,任何一个default实现都不会被继承,你将会获得一个编译时错误。测试

可是,若是你已经把你的类编译过了,那就不会出现编译时错误了。在这一点上,Java 8是不一致的。它有它本身的缘由,有于各类缘由,在这里我不想详细的说明或者是深刻的讨论(由于:版本已经发布了,讨论时间太长,这个平台历来没有这样的讨论)。spa

  • 假如你有两个接口,一个实现类。
  • 其中一个接口实现了一个default方法m()
  • 把接口和实现类一块编译。
  • 修改那个没有包含m()方法的接口,声明m()方法为abstract
  • 单独从新编译修改过的接口。
  • 运行实现类。

上面的状况下类能够正常运行。可是,不能用修改过的接口从新编译,可是用老的接口编译仍然能够运行。接下来命令行

  • 修改那个含有abstract方法m()的接口,建立一个default实现。
  • 编译修改后的接口
  • 运行类:失败。

当两个接口给同一个方法都提供了default实现的时候,这个方法是没法被调用的,除非实现类也实现了这个default方法(要么是直接实现,要么是继承树上更上层的类作实现)。翻译

可是,这个类是兼容的。它能够在使用新接口的状况下被载入,甚至能够执行,只要它没有调用在两个接口中都有default实现的方法。

实例代码

为了演示上面的例子,我给C.java建立了一个测试目录,它下面还有3个子目录,用于存放I1.javaI2.java。测试目录下包含了类C的源码C.java。base目录包含了能够编译和运行的那个版本的接口。I1包含了有default实现的m()方法,I2不包含任何方法。

实现类包含了main方法,因此咱们能够在测试中执行它。它会检查是否存在命令行参数,这样,咱们就能够很方便的执行调用m()和不调用m()的测试。

public class C implements I1, I2 {
    public static void main(String[] args) {
        C c = new C();
        if(args.length == 0 ){
            c.m();
        }
    }
}

public interface I1 {
    default void m(){
        System.out.println("hello interface 1");
    }
}

public interface I2 {
}
复制代码

使用下面的命令行来编译运行:

javac -cp .:base C.java
java -cp .:base C
hello interface 1 
复制代码

compatible目录包含了有abstract方法m()I2接口,和未修改的I1接口。

public interface I2 {
   void m();
} 
复制代码

这个不能用来编译类C:

javac -cp .:compatible C.java
C.java:1: error: C is not abstract and does not override abstract method m() in I2
public class C implements I1, I2 {
       ^
    1 error 
复制代码

错误信息很是精确。由于咱们有前一次编译得到的C.class,若是咱们编译compatible目录下的接口,咱们仍然会获得能运行实现类的两个接口:

javac compatible/I*.java
java -cp .:compatible C
hello interface 1
复制代码

第三个叫作wrong的目录,包含的I2接口也定义了m()方法:

public interface I2 {
    default void m(){
        System.out.println("hello interface 2");
    }
}
复制代码

咱们应该不厌其烦的编译它。尽管m()方法被定义了两次,可是,实现类仍然能够运行,只要它没有调用那个定义了屡次的方法,可是,只要咱们调用m()方法,当即就会失败。这是咱们使用的命令行参数:

javac wrong/*.java
java -cp .:wrong C
    Exception in thread "main" java.lang.IncompatibleClassChangeError: Conflicting
    default methods: I1.m I2.m
        at C.m(C.java)
        at C.main(C.java:5)
java -cp .:wrong C x 
复制代码

结论

当你把给接口添加了default实现的类库移植到Java 8环境下的时候,通常不会有问题。至少Java8类库开发者给集合类添加default方法的时候就是这么想的。使用你类库的应用程序仍然依赖没有default方法的Java7的类库。当使用和修改多个不一样的类库的时候,有很小的概率会发生冲突。如何才能避免呢?

像之前那样设计你的类库。可能依赖default方法的时候不要掉以轻心。万不得已不要使用。明智的选择方法名,避免和其它接口产生冲突。咱们将会学习到Java编程中如何使用这个特性作开发。


欢迎关注知乎专栏《跟上Java8》,分享优秀的Java8中文指南、教程,同时欢迎投稿高质量的文章。

原文连接: javacodegeeks
翻译: ImportNew.com - miracle1919
译文连接: www.importnew.com/10764.html
相关文章
相关标签/搜索