时间:2021-05-02
什么是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实现的时候,这个方法是无法被调用的,除非实现类也实现了这个default方法(要么是直接实现,要么是继承树上更上层的类做实现)。
但是,这个类是兼容的。它可以在使用新接口的情况下被载入,甚至可以执行,只要它没有调用在两个接口中都有 default 实现的方法。
实例代码
为了演示上面的例子,我给 c.java 创建了一个测试目录,它下面还有3个子目录,用于存放 i1.java 和 i2.java 。测试目录下包含了类c的源码 c.java 。base目录包含了可以编译和运行的那个版本的接口。i1包含了有 default 实现的 m() 方法, i2 不包含任何方法。
实现类包含了 main 方法,所以我们可以在测试中执行它。它会检查是否存在命令行参数,这样,我们就可以很方便的执行调用 m() 和不调用 m() 的测试。
? 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 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 { }使用下面的命令行来编译运行:
? 1 2 3 javac -cp .:base c.java java -cp .:base c hello interface 1compatible 目录包含了有 abstract 方法 m() 的 i2 接口,和未修改的 i1 接口。
? 1 2 3 public interface i2 { void m(); }这个不能用来编译类c:
? 1 2 3 4 5 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 目录下的接口,我们仍然会得到能运行实现类的两个接口:
? 1 2 3 javac compatible/i*.java java -cp .:compatible c hello interface 1第三个叫做 wrong 的目录,包含的 i2 接口也定义了 m() 方法:
? 1 2 3 4 5 public interface i2 { default void m(){ system.out.println("hello interface 2"); } }我们应该不厌其烦的编译它。尽管m()方法被定义了两次,但是,实现类仍然可以运行,只要它没有调用那个定义了多次的方法,但是,只要我们调用m()方法,立即就会失败。这是我们使用的命令行参数:
? 1 2 3 4 5 6 7 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编程中如何使用这个特性做开发。
总结
以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,如果有疑问大家可以留言交流,谢谢大家对服务器之家的支持。
原文链接:https://juejin.im/post/5abc9c546fb9a028b61797d8
声明:本页内容来源网络,仅供用户参考;我单位不保证亦不表示资料全面及准确无误,也不保证亦不表示这些资料为最新信息,如因任何原因,本网内容或者用户因倚赖本网内容造成任何损失或损害,我单位将不会负任何法律责任。如涉及版权问题,请提交至online#300.cn邮箱联系删除。
网络的总体布局分为可以比较的、不能比较的、可以消耗自己的和不能消耗自己的。在这两大块中你能做什么和不能做什么将影响你不能做什么。简而言之,每个人都想要无线流量、
4.JavaScript的局限性(JavaScriptLminitations)Q:JavaScript程序不能做什么?A:JavaScript代码不能做下列事
一、PHP能做什么?PHP能做什么?我觉得它很强大,只要我能想到的,它都能做,只是我技术能力还不行╮(╯﹏╰)╭。好吧,一张图,基本了解一下吧(ps:PHP的功
关于什么是用户权限,最简单的定义可能是,“用户能做什么和不能做什么。”在这里,简单的定义就相当不错了。用户的权限分为3类:l登录的权限;l访问特定数据库的权限;
做为在淘宝上刚刚创业的新掌柜经常会出现这样或那样的一些通病!的小编这有一些感受,希望和大家分享!第一、上货不要跟风!要清楚你能做什么,不能做什么。货源是否稳定,