Java 8 新特性之默认方法(Default Methods)

Reading time ~1 minute

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 类的 equalshashCodetoString 方法。
  • 接口中的静态方法必须是 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)
文章作者:凌风
原始连接:https://huangxubo.me/blog/java/java8-default-method/
许可协议:转载请注明原文链接及作者。

CentOS 7 中搭建PPTP、L2TP、IPSec 服务

VPN(Virtual Private Network,即“虚拟私人网络”)是一种通过在数据传输过程中加密的方式来保障数据传输安全的通信方式,L2TP是其中的一种加密方式。 Continue reading

Cnet OS 安装 Shadowscoks-libev 客户端

Published on December 21, 2019

OpenResty 编译安装

Published on December 20, 2019