Java8中的default关键字理解

default关键字介绍

default是在java8中引入的关键字,也可称为Virtual extension methods——虚拟扩展方法。是指,在接口内部包含了一些默认的方法实现(也就是接口中可以包含方法体,这打破了Java之前版本对接口的语法限制),从而使得接口在进行扩展的时候,不会破坏与接口相关的实现类代码。

为什么要有这个特性?

首先,之前的接口是个双刃剑,好处是面向抽象而不是面向具体编程,缺陷是,当需要修改接口时候,需要修改全部实现该接口的类,目前的java8之前的集合框架没有foreach方法,通常能想到的解决办法是在JDK里给相关的接口添加新的方法及实现。然而,对于已经发布的版本,是没法在给接口添加新方法的同时不影响已有的实现。所以引进的默认方法。他们的目的是为了解决接口的修改与现有的实现不兼容的问题。

如何使用default关键字?

1、调用父接口实现

创建接口InterfaceA,并且在接口InterfaceA中定义默认方法helloWorld()

public interface InterfaceA{
    default void helloWorld() {
        System.out.println("hi i'm from InterfaceA");
    }
}

这时可以编写一个类 MyImplementA 实现接口 InterfaceA,并调用接口中定义的默认方法 helloWorld(),代码如下

/**
 * 实现接口InterfaceA
 */
public class MyImplementA implements InterfaceA{
    public static void main(String[] args) {
        MyImplementA myImplement = new MyImplementA();
        //直接调用helloWorld()方法
        myImplement.helloWorld();
    }
}

可以看到,执行的结果为我们之前在接口InterfaceA中定义的默认方法。

QQ20191129-163401@2x.png

此时的类图为

QQ20191129-163711@2x.png

2、同时继承两个接口

创建接口InterfaceB,接口InterfaceB中也定义了默认方法helloWorld()

public interface InterfaceB{
    default void helloWorld() {
        System.out.println("hi i'm from InterfaceB");
    }
}

这时候让类MyImplementAB<.code>同时实现InterfaceAInterfaceB,代码如下:

/**
 * 实现接口InterfaceA,InterfaceB
 */
public class MyImplementAB implements InterfaceA,InterfaceB{
    public static void main(String[] args) {
        MyImplementAB myImplement = new MyImplementAB();
        //直接调用helloWorld()方法
        myImplement.helloWorld();
    }
}

此时会发现编译器报错,报错信息如下:

QQ20191129-164117@2x.png

发生这种情况的原因是,实现类 MyImplementAB 即实现了接口 InterfaceA 又实现了接口 InterfaceB,恰巧两个接口中都定义可相同的默认方法。说白了就是编译器此时已经被干懵了,当我们在MyImplementAB类中调用方法时,它不知道该去调用InterfaceA的默认方法还是去调用InterfaceB的方法。解决方法就是在实现类中实现该方法。代码如下

public class MyImplementAB implements InterfaceA,InterfaceB {

    @Override
    public void helloWorld() {
        System.out.println("hi i'm from MyImplementAB");
    }
    public static void main(String[] args) {
        MyImplementAB myImplement = new MyImplementAB();
        myImplement.helloWorld();
    }
}

执行代码控制台输出如下:

QQ20191129-164332@2x.png

此时的类图为

QQ20191129-164439@2x.png

3、类优先于接口

此时创建一个实现类MyImplementC,该实现类不仅继承了MyImplementA并且实现了InterfaceB

实现类 MyImplementC 的代码如下

public class MyImplementC extends MyImplementAB implements InterfaceB{
    public static void main(String[] args) {
        MyImplementC myImplement = new MyImplementC();
        myImplement.helloWorld();
    }
}

此时的类图为 

QQ20191129-164933@2x.png

此时在实现类MyImplementC中调用helloWorld()方法,到底执行的是MyImplement中的方法还是执行InterfaceB中的方法?

答:因为类优先于接口,所以将会执行MyImplementAB中的方法。执行结果如下 

QQ20191129-164332@2x.png



未经允许请勿转载:程序喵 » Java8中的default关键字理解

点  赞 (2) 打  赏
分享到: