摘要:介绍Java8新增关键字default,它用于在接口中标记方法为默认方法和编写实现逻辑,方便通过新增方法重构接口,而无需修改所有实现类,目的在于兼容接口已有实现类。
综述
default关键字:是在java 8中引入的新概念,也可称为Virtual extension methods——虚拟扩展方法与public、private等都属于修饰符关键字,与其它两个关键字不同之处在于default关键字大部分都用于修饰接口。default修饰方法时只能在接口类中使用,在接口中被default标记的方法可以直接写方法体,而无需修改所有实现了此接口的类。
这打破了Java之前版本对接口的语法限制。之前的版本里 interface 中的方法必须是抽象方法,不能有方法体。现在可以添加 interface 内方法,只需要在方法的前面加一个 default 关键字,表示属于接口内部默认存在的方法,但是,实现类中可以重写其实现,也可以不重写。在调用default修饰的方法时,如果没有匹配的实现类实现它,就执行接口类中定义的默认实现逻辑;否则,执行实现类中的逻辑。
default的前世今生
之前的接口是个双刃剑,好处是面向抽象而不是面向具体编程,缺陷是,当需要修改接口时候,需要修改该接口的全部实现类。例如,java8之前的集合框架没有foreach方法,通常能想到的解决办法是在JDK里给相关的接口添加新的方法及实现。然而,对于已经发布的版本,是没法在给接口添加新方法的同时不影响已有的实现。所以引进了默认方法。他们的目的是为了解决实现接口的缺陷问题(就是如果想修改接口时,所有实现了该接口的类都需要去修改)。
通俗地讲,不使用default定义方法的话,该interface的实现类都必须重写所有的抽象方法,但是,在实现含有default方法的接口时,可以直接使用接口的default方法,也可以在个别实现类中重写接口中的default方法从而实现自己的业务逻辑。
如何使用default关键词
实现类会继承接口中的default方法。创建一个person类, 使用default修饰该方法:
public interface Person {
default String getName(){
return "我是默认姓名";
}
default String getId(){
return "我是默认ID";
}
}
实现该接口的类不需要去实现默认方法,也可以实现,我们实现其中的一个,看看效果:
@Service
public class Student implements Person {
@Override
public String getId() {
return "32";
}
public static void main(String[] args) {
Student student = new Student();
System.out.println(student.getName());
System.out.println("重写接口类的默认方法,则结果由实现类决定:");
System.out.println(student.getId());
}
}
执行main方法进行验证,运行结果如下:
我是默认姓名
重写接口类的默认方法,则结果由实现类决定:
32
总而言之,当我们修改interface类的代码添加新功能时,必须要想尽办法兼容历史版本的代码,更何况这还是伟大的java,default方法应运而生。
解决冲突
如果集成的两个接口有default定义的同一个方法,例如
public interface Person1 {
default String getName(){
return "我是另一个默认姓名";
}
}
则编译器会报错。发生这种情况的原因是,实现类MyImplement即实现了接口Person又实现了接口Person1,恰巧两个接口中都定义可相同的默认方法。说白了就是编译器此时已经被干懵了,当我们在实现类Student中调用方法时,它不知道该去调用哪个接口类的默认方法。解决方法就是在实现类中实现该方法,重写接口实现。为什么要重写方法呢?是因为类在继承接口中的default方法时,不知道应该继承哪一个接口中的default方法。修改Student类后,代码如下:
@Service
public class Student implements Person, Person1 {
@Override
public String getName() {
return "我是实现类中定义的姓名";
}
@Override
public String getId() {
return "32";
}
public static void main(String[] args) {
Student student = new Student();
System.out.println(student.getName());
System.out.println("重写接口类的默认方法,则结果由实现类决定:");
System.out.println(student.getId());
}
}
类优先于接口
创建一个实现类Student1,该实现类不仅继承了Student而且实现了Person1。
public class Student1 extends Student implements Person1 {
public static void main(String[] args) {
Student1 student = new Student1();
System.out.println(student.getName());
}
}
问题:在实现类Student1中调用 getName() 方法,到底执行的是Student还是Person1中的方法?
答:因为类优先于接口,所以将会执行Student中的方法。
结束语
本文介绍Java8新增关键字default,它用于在接口中标记方法为默认方法和编写方法体,方便通过新增方法重构接口,而无需修改所有实现类,目的在于兼容接口已有实现类。同时,演示了类优先于接口的规则。
大家如果对于本文介绍的内容有不理解的地方,请在评论区留言。如果碰到【相关】问题也可以在评论区留言,和诸位互动交流。三人行必有我师,一起进步。大家可随意尽情享用本文啦!