java接口默认方法

原文链接:https://liushiming.cn/2020/02/23/java-default-methods/

概述

Java8带来了一些全新的特性,包括lambda表达式、函数接口、方法引用、流、可选方法、接口中的静态方法和默认方法。

在本文中,我们将深入讨论为什么java8接口新增了默认方法,如何使用默认方法,并讨论一些有用的用例。

默认方法

在接口中用default关键字定义接口的默认方法。普通接口方法是不能有实现的,默认方法必须有实现:

public interface MyInterface {
     
    // 普通接口方法
     
    default void defaultMethod() {
        // 默认方法
    }
}

为什么需要默认方法

在Java8之前,接口只能有抽象方法。如果不强制所有实现类创建新方法的实现,就不能向现有接口添加新功能。

Java8新增默认方法的原因非常明显。

在一个典型的基于抽象的设计中,一个接口有一个或多个实现类。接口要新增方法,所有的实现都要新增这个方法的实现。否则就不满足接口的约束。

默认接口方法是处理此问题的解决方案。在接口添加默认方法不需要修改实现类,接口新增的默认方法在实现类中直接可用。

默认方法的使用

定义MobilePhone接口,其中setTime,getLengthInCm为默认方法

interface MobilePhone {
    /**
     * 获取手机品牌
     */
    String getBrand();

    /**
     * 获取手机颜色
     */
    String getColor();

    /**
     * 获取手机长度(毫米)
     */
    Double getLength();

    /**
     * 设置手机时间
     */
    default String setTime(String newTime) {
        return "time set to " + newTime;
    }

    /**
     * 对getLength方法进行拓展,返回厘米为单位的长度
     */
    default String getLengthInCm() {
        return getLength() / 10 + "cm";
    }
}

默认方法在实现类中可以直接使用

public class DefaultTests implements MobilePhone {
    @Override
    public String getBrand() {
        return "iphone";
    }

    @Override
    public String getColor() {
        return "red";
    }

    @Override
    public Double getLength() {
        return 150.00;
    }

    @Test
    public void defaultTest() {
        System.out.println(setTime("8:00 am"));
        System.out.println(getLengthInCm());
    }
}

结果:

time set to 8:00 am
15.0 cm

如果在某个时候我们往接口添加更多的默认方法,实现类可以不用修改继续使用

默认方法的最典型用法是逐步为接口提供附加功能,而不破坏实现类。

此外,它们还可以用来为现有的抽象方法提供额外的功能:

interface MobilePhone {
    /**
     * 获取手机长度(毫米)
     */
    Double getLength();

    /**
     * 对getLength方法进行拓展,返回厘米为单位的长度
     */
    default String getLengthInCm() {
        return getLength() / 10 + "cm";
    }
}

默认方法的多继承

Apple接口和Samsung接口继承MobilePhone接口:

interface Apple extends MobilePhone {
    @Override
    default String setTime(String newTime) {
        return "time set to " + newTime + " in apple";
    }
}

interface Samsung extends MobilePhone {
    @Override
    default String setTime(String newTime) {
        return "time set to " + newTime + " in samsung";
    }
}

DefaultTests实现Apple和Samsung接口,必须对setTime方法进行重写,否则对象将不知道该使用Apple的setTime方法还是Samsung的setTime方法,因为它们同名了

public class DefaultTests implements Apple, Samsung {
    @Override
    public String getBrand() {
        return "iphone";
    }

    @Override
    public String getColor() {
        return "red";
    }

    @Override
    public Double getLength() {
        return 150.00;
    }


    @Override
    public String setTime(String newTime) {
        return Apple.super.setTime(newTime);
    }

    @Test
    public void defaultTest() {
        System.out.println(setTime("8:00 am"));
        System.out.println(getLengthInCm());
    }
}

结果

time set to 8:00 am in apple
15.0 cm

总结

在本文中,我们深入探讨了Java8中接口默认方法的使用。乍一看,这个特性可能有点马虎,特别是纯粹从面向对象的角度来看。理想情况下,接口不应该封装行为,而应该只用于定义特定类型的公共API。

但是在维护现有代码的向后兼容性时,静态方法和默认方法是一种很好的折衷。

参考资料

Baeldung - Static and Default Methods in Interfaces in Java

posted @ 2020-03-01 16:16  huahuayu  阅读(9770)  评论(0编辑  收藏  举报