第十一节 Java8的接口默认方法

亮子 2021-09-14 21:17:18 10906 0 0 0

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) {
        Collections.sort(this, c);
    }
}

可以看到,这个新增的sort方法有方法体,由default修饰符修饰,这就是接口的默认方法

很显然,默认方法不是static的,所以必须由接口的实现类的实例来调用这些默认方法。

我们知道Java语言中一个类只能继承一个父类,但是一个类可以实现多个接口。随着默认方法在Java8中的引入,有可能出现一个类继承了多个签名一样的方法。这种情况下,类会选择使用哪一个函数呢?

为解决这种多继承关系,Java8提供了下面三条规则:

  1. 类中的方法优先级最高,类或父类中声明的方法的优先级高于任何声明为默认方法的优先级。

  2. 如果第一条无法判断,那么子接口的优先级更高:方法签名相同时,优先选择拥有最具体实现的默认方法的接口, 即如果B继承了A,那么B就比A更加具体。

  3. 最后,如果还是无法判断,继承了多个接口的类必须通过显式覆盖和调用期望的方法, 显式地选择使用哪一个默认方法的实现。

图片alt

1)、场景1

interface A {
    default void hello() {
        System.out.println("hello from A");
    }
}

interface B extends A {
    @Override
    default void hello() {
        System.out.println("hello from B");
    }
}

public class DefaultInterfaceMethod  implements A, B{

    public static void main(String[] args) {
        new DefaultInterfaceMethod().hello();
    }
}

2)、场景2:

interface AA {
    default void hello() {
        System.out.println("hello from AA");
    }
}

interface BB extends AA {
    @Override
    default void hello() {
        System.out.println("hello from BB");
    }
}

class DD implements AA {
    @Override
    public void hello() {
        System.out.println("hello from DD");
    }
}

public class DefaultInterfaceMethod2 extends DD  implements A, B{

    public static void main(String[] args) {
        new DefaultInterfaceMethod2().hello();
    }
}