什么是默认方法-Default Methods
简单的说,就是可以在接口中定义一个已实现方法,且该接口的实现类不需要实现该方法;
如下示例:
interface GreetingService { void sayMessage(String message); //可以在接口中定义默认方法 default void sayHello(){ System.out.println("Hello"); } } //实现类不需要实现接口中的默认方法 class GreetingServiceImpl implements GreetingService{ @Override public void sayMessage(String message) { } }
为什么要有默认方法
主要是为了方便扩展已有接口;如果没有默认方法,加入给JDK中的某个接口添加一个新的抽象方法,那么所有实现了该接口的类都得修改,影响将非常大。
使用默认方法,可以给已有接口添加新方法,而不用修改该接口的实现类。当然,接口中新添加的默认方法,所有实现类也会继承该方法。
举个例子,在Java 8的Iterable接口中,新增了一个默认方法forEach,也正因为forEach是默认方法,才不用修改所有Iterable接口的实现类。
Iterable接口新增的forEach方法如下(入参是一个函数式接口,因此支持Lambda表达式):
default void forEach(Consumer<? super T> action) { Objects.requireNonNull(action); for (T t : this) { action.accept(t); } }
因为Collection接口继承了Iterable接口,所以我们可以在集合类中使用forEach方法,如下,这里使用了方法引用(一种更加紧凑的Lambda表达式)
List<String> list = new ArrayList<String>(); list.add("001"); list.add("002"); list.forEach(System.out::println);
可见,我们在未破坏Iterable接口实现类的前提下,给Iterable接口的所有实现类添加了一个新方法forEach,这在Java 8之前是不可能的。
重写Override默认方法
- 如果子类没有重写父接口默认方法的话,会直接继承父接口默认方法的实现;
- 如果子类重写父接口默认方法为普通方法,则与普通方法的重写类似;
- 如果子类(接口或抽象类)重写父接口默认方法为抽象方法,那么所有子类的子类需要实现该方法;
关于默认方法调用冲突
因为一个类是可以实现多个接口的,如果多个接口定义了同样的默认方法,那么子类如何调用父类的默认方法呢?
具体调用流程如下:
1、首先,如果子类覆盖了父类的默认方法,那么什么也不用想,直接使用调用子类覆盖后的方法;
2、其次,优先选择调用更加具体的接口默认方法,什么意思呢,举个例子,如果A1接口继承A接口,那么A1接口相对A接口就更加具体,当C类实现了A1接口的时候,就优先调用A1接口的默认方法;
3、最后,如果C类同时实现A1接口和A2接口,且A1和A2有同名的默认方法,那么选择哪个接口的默认方法呢?答案是编译器报错,提示定义了重名的方法,快速修复方式是覆盖其中的一个即可;
关于这块内容,在网上看到一段有意思的代码,如下,知道为什么会报错吗?如果删除InterfaceB中的foo方法,是否还会报错?往InterfaceC中添加foo方法又会怎样?
参考资料
https://docs.oracle.com/javase/tutorial/java/IandI/defaultmethods.html
https://dzone.com/articles/interface-default-methods-java