Java8是Oracle于2014年3月发布的一个重要版本,其API在现存的接口上引入了非常多的新方法。
例如,Java8的List接口新增了sort方法。在Java8之前,则每个实现了List接口的类必须定义sort方法的实现,或者从父类中继承它的实现。想象一下,如果List接口的继承体系非常庞杂,那么整个集合框架的维护量有多么大!
为此,在Java8中引入了一种新的机制:接口支持申明带实现的方法。
默认方法
前文提到了Java8中List接口新增了sort方法,其源码如下:
public interface List<E> extends Collection<E> {
// ...其他成员
default void sort(Comparator<? super E> c) {
...
...
}
}
可以看到,这个新增的sort方法有方法体,由default修饰符修饰,这就是接口的默认方法。
很显然,默认方法不是static的,所以必须由接口的实现类的实例来调用这些默认方法。
下面自定义一个接口,练习使用默认方法。
public interface Sized {
// 普通抽象方法,默认是public abstract修饰的,没有方法体
int size();
/*
* 默认方法,有方法体
* 任何一个实现了Sized接口的类都会向动继承isEmpty的实现
*/
default boolean isEmpty() {
return this.size() == 0;
}
}
其实,随着JDK版本的不断升级,API在不断演进,默认方法在Java8的API中已经大量地使用了,上面List接口中的sort方法就是其中一个。
和抽象类的区别
有同学可能发现了,Java8中加入了默认方法的接口,这不就是以前的抽象类吗?其实,两者还是有区别的。
解决冲突
我们知道Java语言中一个类只能继承一个父类,但是一个类可以实现多个接口。随着默认方法在Java8中的引入,有可能出现一个类继承了多个签名一样的方法。这种情况下,类会选择使用哪一个函数呢?
为解决这种多继承关系,Java8提供了下面三条规则:
让我们一起看几个例子 。
场景1:
public interface A {
default void hello() {
System.out.println("hello from A");
}
}
public interface B extends A {
default void hello() {
System.out.println("hello from B");
}
}
public class C implements A, B {
public static void main(String[] args) {
new C().hello();
}
}
如图1,是这个场景的UML图。
我们对照上面三条规则来看,类C中main()方法会输出什么?
场景2:
如果C像下面这样继承了D,会怎么样?
public class D implements A {
}
public class C extends D implements A, B {
public static void main(String[] args) {
new C().hello();
}
}
如图2,是这个场景的UML图。
同样,我们对照着三条规则来看:
场景3:
将上面的D稍作修改:
public class D implements A {
public void hello() {
System.out.println("hello from D");
}
}
结果又如何?
由于依据规则(1),父类中声明的方法具有更高的优先级,所以程序会打印输出”hello from D”。
场景4:
假设现在B不在继承A:
public interface A {
default void hello() {
System.out.println("hello from A");
}
}
public interface B {
default void hello() {
System.out.println("hello from B");
}
}
public class C implements A, B {
public static void main(String[] args) {
new C().hello();
}
}
如图3,是这个场景的UML图。
此时,由于编译器无法识别A还是B的实现更加具体,所以会抛出编译错误:”C inherits unrelated defaults for hello() from types A and B“。
像这种场景要解决冲突,可以在C中覆盖hello()方法并在方法内显示的选择调用A还是B的方法。
调用方式如下:
public class C extends D implements A, B {
public void hello() {
// 显式地选择调用接口B中的方法
// 同理,要调用接口A中的方法,可以这样:A.super.hello()
B.super.hello();
}
public static void main(String[] args) {
// 输出 hello from B
new C().hello();
}
}
场景5:
public interface A {
default void hello() {
System.out.println("hello from A");
}
}
public interface B extends A{
}
public interface C extends A{
}
public class D implements B, C {
public void hello() {
new D().hello();
}
}
此时,只有一个方法申明可以选择,所以程序会输出”hello from A”。
欢迎工作一到五年的Java工程师朋友们加入Java程序员开发: 721575865
群内提供免费的Java架构学习资料(里面有高可用、高并发、高性能及分布式、Jvm性能调优、Spring源码,MyBatis,Netty,Redis,Kafka,Mysql,Zookeeper,Tomcat,Docker,Dubbo,Nginx等多个知识点的架构资料)合理利用自己每一分每一秒的时间来学习提升自己,不要再用”没有时间“来掩饰自己思想上的懒惰!趁年轻,使劲拼,给未来的自己一个交代!
有时候你会觉得很轻松很快乐,但有时候又会觉得好难,好像看不到希望。但我们必须接受生活似过山车的这个事实。不必羡慕别人,也不埋怨自己,脚踏实地,默默努力,做好该做的,剩下的就交给时间。 我今天在我们圈子里听到的一个同行的分享。它的很多客户呢,都是批量购买,甚至高峰时期会变成老客户反复购买,这种生意这...
很多朋友都在上找项目,有的朋友也被割过韭菜,因为做互联网的人没有谁是没被割过韭菜的,有的韭菜终有一天会茁壮成长,有的韭菜肯定会被无情的镰刀割到你怀疑人生。 韭菜就是被割的,你不割它会烂在地里。韭菜内心愤愤不平,继续辗转到下一块地再次茁壮生长,殊不知,山顶、山腰、山谷都可以割,而且镰刀更加锋利。...
今天面试了一位00后应届毕业女生,感觉真的和以前的90后求职者不一样,进来办公室就把包往桌子上一丢,用手挽了挽头发,轻松的说了句:“不好意思,来晚了,我们可以开始了。”我颇感诧异,这气势上就给人一种统治力和压迫感,高挑的身材,带着口罩,双眼化了妆,看起来颜值很高。 她把简历放在桌子上,用手拍了...
随着天气的变冷,很多友友都开始为自己置办加绒的衣服,而近期一种名为奥粒绒的面料便成了各大直播间的主推,不过由于该面料和之前的摇粒绒外观有一点点的相似,下面小编就和大家一起看看奥粒绒保暖效果怎么样。奥粒绒和摇粒绒一样吗奥粒 随着天气的变冷,很多友友都开始为自己置办加绒的衣服,而近期一种名为奥粒绒的面...
金昌市永昌县旅游景点 金川峡水库-丝绸古道明亮的明珠景点一: 金昌市十大著名旅游景点? 金川峡水库位于盛荣风景区入口处。它以永昌县2000多只春天形成的金川河命名。它不仅是金昌市的一级水源保护区,也是真正的鸟类天堂。湖光倒影,风景迷人,生态优美,魅力无穷!白天鹅、大白鹭、绿头鸭、秋沙...
口碑最好十大防晒伞排名 在导致皮肤老化的外源性因素中,紫外线是导致皮肤老化的最大原因,因此防晒霜更为重要。每个季节都要注意防晒霜。现在市场上有很多种防晒霜,那么哪种品牌的防晒霜最好?下面将列出世界十大防晒霜。名单上有什么? 全球防晒霜排行榜十强? Cosme Decorte多重防晒乳...
破壁机怎么打豆浆步骤 当许多朋友使用破壁机制作豆浆时,他们发现会有很多泡沫。这是因为大豆中的蛋白质与水结合形成蛋白质胶体,增加了水的张力。在破壁机的不断搅拌过程中,产生了大量的泡沫。这些泡沫中含有蛋白质胶体,因此不容易破裂。本文主要研究泡沫的原理。 破壁机打豆浆沫太多? 当谈到豆...
红薯是生活中比较常见的一种美食,很多人都会煮着吃,味道很不错,深受人们喜欢。红薯要吃熟透的,没熟的红薯不容易消化吸收,会容易导致积食,引起肠胃不适,最好放锅里面弄熟之后再吃。红薯没熟透可以吃吗虽然可以吃,但是不建议过多的 红薯是生活中比较常见的一种美食,很多人都会煮着吃,味道很不错,深受人们喜欢。...
鸭蛋保存方法和注意事项,鸭蛋的保存方法介绍。小编带你了解更多相关信息。 1、如果是夏天比较热的时候,可以放进冰箱冷藏室里存放,注意不要冷冻,冻过的鸭蛋很不好吃。一般放在冰箱上层冷藏室就行了。可以把熟鸭蛋做真空处理,能保证鸭蛋放比较长的时间。这需要专门的仪器,如果鸭蛋比较多可去附近加工厂做真空处...
水果保存方法和注意事项,水果保存方法介绍。小编带你了解更多相关信息。 1、购买时令水果,时令水果多半新鲜、品质佳,而且价钱又平实。不论何种水果,果实饱满、大小适中(表示果实发育完全)、外形完好、无碰伤及病斑等,都是基本的选择要点。而果实拿上手沉甸甸、具重量感,通常表示水分含量多,吃起来应是“香...
买的酒能不能带上公交车,买的酒能否带上公交车。小编带你了解更多相关信息。 1、乘坐公交车可以带酒,但是有携带的条件,必须是低度未开封、有纸盒或塑料盒包装的白酒才可以带上公交车,但是在乘车过程中严禁打开,以免发生危险情况。 2、白酒(外文名:是中国酒类(除了果酒、米酒外)的统称,又称烧酒、老白...
红薯的吃法有很多,烤着吃和蒸着吃是其中最常见的两种吃法,很多人都会在家烤红薯或者蒸红薯吃。烤红薯的热量和蒸红薯的热量是不一样的,相对比而言,烤红薯的热量要比蒸红薯的热量高一点。烤红薯和蒸红薯的热量是一样的吗不一样,烤红薯 红薯的吃法有很多,烤着吃和蒸着吃是其中最常见的两种吃法,很多人都会在家烤红薯...
柚子是生活中比较常见的一种水果,它含有多种维生素、矿物质以及膳食纤维等,适量食用对身体有一定好处,很多人都会经常吃柚子。柚子有红柚子和白柚子,相对比而言,红柚子和白柚子的热量是差不多的,几乎是一样的。红柚子和白柚子热量一 柚子是生活中比较常见的一种水果,它含有多种维生素、矿物质以及膳食纤维等,适量...
红薯的吃法比较多,烤红薯就是其中比较常见的一种吃法,它的味道香甜可口,吃起来软软糯糯的,营养价值高,深受人们喜欢。烤红薯最好当天吃掉,吃不完的烤红薯可以密封冷藏,第二天还是可以吃的,但是味道和口感可能会变差。烤红薯吃不完 红薯的吃法比较多,烤红薯就是其中比较常见的一种吃法,它的味道香甜可口,吃起来...
红薯是生活中比较常见的一种粗粮食物,很多人都会在家烤红薯吃。烤红薯一般都是现做现吃的,如果烤红薯吃了一半,吃不完了,可以密封冷藏保存到第二天吃,但是第二天吃需要重新加热,并且味道和口感也会变差。烤红薯吃了一半第二天能吃吗 红薯是生活中比较常见的一种粗粮食物,很多人都会在家烤红薯吃。烤红薯一般都是现...