温馨提示×

温馨提示×

您好,登录后才能下订单哦!

密码登录×
登录注册×
其他方式登录
点击 登录注册 即表示同意《亿速云用户服务条款》

Java 8中default方法有什么用

发布时间:2021-07-16 10:29:55 来源:亿速云 阅读:169 作者:小新 栏目:编程语言

这篇文章主要介绍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 。

  • 单独重新编译修改过的接口。

  • 运行实现类。

Java 8中default方法有什么用

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

  • 修改那个含有 abstract 方法 m() 的接口,创建一个 default 实现。

  • 编译修改后的接口

  • 运行类:失败。

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

Java 8中default方法有什么用

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

实例代码

Java 8中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

以上是“Java 8中default方法有什么用”这篇文章的所有内容,感谢各位的阅读!希望分享的内容对大家有帮助,更多相关知识,欢迎关注亿速云行业资讯频道!

向AI问一下细节

免责声明:本站发布的内容(图片、视频和文字)以原创、转载和分享为主,文章观点不代表本网站立场,如果涉及侵权请联系站长邮箱:is@yisu.com进行举报,并提供相关证据,一经查实,将立刻删除涉嫌侵权内容。

AI