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)
作者:凌风
**粗体** _斜体_ [链接](http://example.com) `代码` - 列表 > 引用
。你还可以使用@
来通知其他用户。