原文链接: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。
但是在维护现有代码的向后兼容性时,静态方法和默认方法是一种很好的折衷。