Java 8中default方法能做什么?不能做什么?

什么是default方法?

Java 8 发布以后,可以给接口添加新方法,但是,接口仍然可以和它的实现类保持兼容。这非常重要,因为你开发的类库可能正在被多个开发者广泛的使用着。而Java 8之前,在类库中发布了一个接口以后,如果在接口中添加一个新方法,那些实现了这个接口的应用使用新版本的接口就会有崩溃的危险。

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

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

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

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

多继承的时候呢?

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

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

但是,如果你已经把你的类编译过了,那就不会出现编译时错误了。在这一点上,Java 8是不一致的。它有它自己的原因,有于各种原因,在这里我不想详细的说明或者是深入的讨论(因为:版本已经发布了,讨论时间太长,这个平台从来没有这样的讨论)。

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

上面的情况下类可以正常运行。但是,不能用修改过的接口重新编译,但是用老的接口编译仍然可以运行。接下来

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

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

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

实例代码

为了演示上面的例子,我给 C.java 创建了一个测试目录,它下面还有3个子目录,用于存放 I1.java 和 I2.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编程中如何使用这个特性做开发。

总结

以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,如果有疑问大家可以留言交流,谢谢大家对我们的支持。

(0)

相关推荐

  • Java8中的default方法详解

    Java 8新增了default方法,它可以在接口添加新功能特性,而且还不影响接口的实现类.下面我们通过例子来说明这一点. 复制代码 代码如下: public class MyClass implements InterfaceA {  public static void main(String[] args){  }    @Override  public void saySomething() {   // TODO Auto-generated method stub  } } int

  • Java接口默认方法带来的问题分析【二义性问题】

    本文实例分析了Java接口默认方法带来的问题.分享给大家供大家参考,具体如下: 一 点睛 Java 8中,如果一个类实现两个或多个接口,即"变相"的多继承,但是若其中两个接口都包含一个名字相同的default方法,会出现二义性问题. 二 实战 1 代码 interface faceA //定义接口faceA { void someMethod( ); default public void DefaultMethod( )//定义接口中默认方法 { System.out.println

  • Java8的default方法详细介绍

    什么是default方法? Java 8发布以后,可以给接口添加新方法,但是,接口仍然可以和它的实现类保持兼容.这非常重要,因为你开发的类库可能正在被多个开发者广泛的使用着.而Java 8之前,在类库中发布了一个接口以后,如果在接口中添加一个新方法,那些实现了这个接口的应用使用新版本的接口就会有崩溃的危险. 有了Java 8,是不是就没有这种危险了?答案是否定的. 给接口添加default方法可能会让某些实现类不可用. 首先,让我们看下default方法的细节. 在Java 8中,接口中的方法可

  • Java带default方法接口的应用示例

    本文实例讲述了Java带default方法接口的应用.分享给大家供大家参考,具体如下: 一 带default方法接口的实现 1 代码 interface InterfaceA // 定义一个接口 { public static String INFO = "static final." ; // 全局常量 public void print() ; // 抽象方法 default public void otherprint() // 带方法体的默认方法 { System.out.pr

  • 深入解析Java接口(interface)的使用

    Java接口(interface)的概念及使用 在抽象类中,可以包含一个或多个抽象方法:但在接口(interface)中,所有的方法必须都是抽象的,不能有方法体,它比抽象类更加"抽象". 接口使用 interface 关键字来声明,可以看做是一种特殊的抽象类,可以指定一个类必须做什么,而不是规定它如何去做. 现实中也有很多接口的实例,比如说串口电脑硬盘,Serial ATA委员会指定了Serial ATA 2.0规范,这种规范就是接口.Serial ATA委员会不负责生产硬盘,只是指定

  • java自定义注解接口实现方案

    java注解是附加在代码中的一些元信息,用于一些工具在编译.运行时进行解析和使用,起到说明.配置的功能. 注解不会也不能影响代码的实际逻辑,仅仅起到辅助性的作用.包含在 java.lang.annotation 包中. 1.元注解 元注解是指注解的注解.包括 @Retention @Target @Document @Inherited四种. 1.1.@Retention: 定义注解的保留策略 Java代码 复制代码 代码如下: @Retention(RetentionPolicy.SOURCE

  • 详解Java8新特性之interface中的static方法和default方法

    为什么要单独写个Java8新特性,一个原因是我目前所在的公司用的是jdk8,并且框架中用了大量的Java8的新特性,如上篇文章写到的stream方法进行过滤map集合.stream方法就是接口Collection中的default方法.所以准备专门写写关于java8新特性的文章,虽然现在10已经发布了.但还是要认真的去了解下新版本的变化. static方法 java8中为接口新增了一项功能:定义一个或者更多个静态方法.用法和普通的static方法一样. 代码示例 public interface

  • Java接口和抽象类用法实例总结

    本文实例讲述了Java接口和抽象类用法.分享给大家供大家参考,具体如下: 接口 1 因为java不支持多重继承,所以有了接口,一个类只能继承一个父类,但可以实现多个接口,接口本身也可以继承多个接口. 2 接口里面的成员变量默认都是public static final类型的.必须被显示的初始化. 3 接口里面的方法默认都是public abstract类型的.隐式声明. 4 接口没有构造方法,不能被实例化. 5 接口不能实现另一个接口,但可以继承多个接口. 6 类如果实现了一个接口,那么必须实现

  • java中接口(interface)及使用方法示例

    1.接口:一种把类抽象的更彻底,接口里只能包含抽象方法的"特殊类".接口不关心类的内部状态数据,定义的是一批类所遵守的规范.(它只规定这批类里必须提供某些方法,提供这些方法就可以满足实际要求). 在JAVA编程语言中是一个抽象类型,是抽象方法的集合,接口通常以interface来声明.一个类通过继承接口的方式,从而来继承接口的抽象方法. 接口并不是类,编写接口的方式和类很相似,但是它们属于不同的概念.类描述对象的属性和方法.接口则包含类要实现的方法. 除非实现接口的类是抽象类,否则该类

  • Java基础教程之封装与接口

    总结之前的内容,对象(object)指代某一事物,类(class)指代象的类型.对象可以有状态和动作,即数据成员和方法. 到现在为止,数据成员和方法都是同时开放给内部和外部的.在对象内部,我们利用this来调用对象的数据成员和方法.在对象外部,比如当我们在另一个类中调用对象的时,可以使用 对象.数据成员 和 对象.方法() 来调用对象的数据成员和方法. 我们将要封装(encapsulation)对象的成员(成员包括数据成员和方法),从而只允许从外部调用部分的成员.利用封装,我们可以提高对象的易用

随机推荐