Java 8 默认方法-Default Methods

阅读目录

  • 什么是默认方法-Default Methods
  • 为什么要有默认方法
  • 重写Override默认方法
  • 关于默认方法调用冲突
  • 参考资料
 

什么是默认方法-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方法又会怎样?

Java 8 默认方法-Default Methods

猜你喜欢

转载自www.linuxidc.com/Linux/2016-09/135436.htm