1. 概述
Java 8 引入了新的语言特性——默认方法(Default Methods)。
默认方法允许您添加新的功能到现有库的接口中,并能确保与采用旧版本接口编写的代码的二进制兼容性。
1.1 为什么要有默认方法
在 Java 8 之前,接口与其实现类之间的耦合度太高了,当需要为一个接口添加方法时,所有的实现类都必须随之修改。默认方法解决了这个问题,它可以为接口添加新的方法,而不会破坏已有的接口的实现。这在 lambda 表达式作为 Java 8 语言的重要特性而出现之际,为升级旧接口且保持向后兼容提供了途径。
还有就就是 Java 8 的函数式接口只允许有一个抽象方法,但可以有多个默认方法。
String[] array = new String[] {“hello”,”, “,”world”,};
List<String> list = Arrays.asList(array);
list.forEach(System.out::println); // 这是 jdk 1.8 新增的接口默认方法
这个 forEach 方法是 jdk 1.8 新增的接口默认方法,正是因为有了默认方法的引入,才不会因为 Iterable 接口中添加了 forEach 方法就需要修改所有 Iterable 接口的实现类。
1.2 语法格式
interface InterfaceA {
default void print() {
System.out.println(“InterfaceA print”);
}
}
2. 默认方法(default)
2.1 实例
interface InterfaceA {
default void print() {
System.out.println(“InterfaceA print”);
}
}
class ClassA implements InterfaceA {
}
public class Java8Test {
public static void main(String[] args) {
new ClassA().print(); // 打印:“InterfaceA print”
}
}
ClassA 类并没有实现 InterfaceA 接口中的 print 方法,InterfaceA 接口中提供了 print 方法的默认实现,因此可以直接调用 ClassA 类的 print 方法。
2.2 默认方法的继承
interface InterfaceA {
default void print() {
System.out.println(“InterfaceA print”);
}
}
interface InterfaceB extends InterfaceA {
}
interface InterfaceC extends InterfaceA {
@Override
default void print() {
System.out.println(“InterfaceC print”);
}
}
interface InterfaceD extends InterfaceA {
@Override
void print();
}
public class Java8Test {
public static void main(String[] args) {
new InterfaceB() {}.print(); // 打印:”InterfaceA print”
new InterfaceC() {}.print();// 打印:”InterfaceC print”
new InterfaceD() {
@Override
public void print(){
System.out.println(“InterfaceD print”);
}
}.print();// 打印:“InterfaceD print”
// 或者使用 lambda 表达式
((InterfaceD) () -> System.out.println(“InterfaceD print”)).print();
}
}
接口默认方法的继承分三种情况(分别对应上面的 InterfaceB 接口、InterfaceC 接口和 InterfaceD 接口):
不覆写默认方法,直接从父接口中获取方法的默认实现。
覆写默认方法,这跟类与类之间的覆写规则相类似。
覆写默认方法并将它重新声明为抽象方法,这样新接口的子类必须再次覆写并实现这个抽象方法。
2.3 类优先原则
接口默认方法的类优先原则
若一个接口中定义了一个默认方法,而另外一个父类或接口中又定义了一个同名的方法时
选择父类中的方法。如果一个父类提供了具体的实现,那么接口中具有相同名称和参数的默认方法会被忽略。
接口冲突。如果一个父接口提供一个默认方法,而另一个接口也提供了一个具有相同名称和参数列表的方法(不管方法是否是默认方法),那么必须覆盖该方法来解决冲突
public interface FunA {
default String getName() {
return “aaaa”;
}
}
public class ClassA {
public String getName() {
return “bbb”;
}
}
public class SubClass extends ClassA implements FunA {
}
测试方法
@Test
public void t1(){
SubClass subClass = new SubClass();
System.out.println(subClass.getName()); // 输出的是 bbb
}
注意 :
default 关键字只能在接口中使用(以及用在 switch 语句的 default 分支),不能用在抽象类中。
接口默认方法不能覆写 Object 类的 equals、hashCode 和 toString 方法。
接口中的静态方法必须是 public 的,public 修饰符可以省略,static 修饰符不能省略。
即使使用了 java 8 的环境,一些 IDE 仍然可能在一些代码的实时编译提示时出现异常的提示(例如无法发现 java 8 的语法错误),因此不要过度依赖 IDE。
3. 静态默认方法
Java 8 的另一个特性是接口可以声明(并且可以提供实现)静态方法
interface InterfaceA {
default void print() {
System.out.println(“InterfaceA print”);
}
static void staticMethod(){
System.out.println(“InterfaceA staticMethod”);
}
}
本文首发于凌风博客:Java 8 默认方法 (Default Methods) 作者:凌风