Java 8 中函数接口,陈货翻新了炒,只是为了Lambda表达式


Java开发人员应该对java.lang.Runnable,java.util.Comparator,java.util.concurrent.Callable 等等接口不会感到陌生。他们都只有一个单一的抽象方法。这样的接口,我们通常叫单一抽象方法接口(SAMSingle Abstract Method Interface)。

以前大家应该经常使用下面的代码片段

public class InnerAnonymousClassSample {

    public static void main(String[] args){


        new Thread(new Runnable() {
            @Override
            public void run() {
                System.out.println("Anonymous Class Thread Demo");
            }
        }).start();

    }}

Java 8 中,对于这种拥有单一抽象方法的接口,换了个名字,叫函数接口。所以,这个不是什么新东西,这个名字也是为了Lambada表达式而生。还是拿Runnable来说事,看看和Lambda表达式合壁后的结果。

public class ThreadWithLambda {

    public static void main(String[] args){
        new Thread(() -> System.out.println("Thread with Lambda expression Demo")).start();


    }
}

回到函数接口上来,Java引入了@FunctionalInterface注解。用了它,你定义的接口就得是只有一个抽象方法的函数接口,否则会报编译错误。


@FunctionalInterface
public interface FunInterfaceAnnotationDemo {

        public void absMethodDemo();
        public void absMoreMethodDemo();

}

Error:(8, 1) java: 意外的 @FunctionalInterface 注释
        com.tr.learning.lambda.FunInterface.FunInterfaceAnnotationDemo 不是函数接口
        在 接口 com.tr.learning.lambda.FunInterface.FunInterfaceAnnotationDemo 中找到多个非覆盖抽象方法

上述方法可以通过Java 8中引入的关键字default 在接口中为其中一个抽象方法指定一个默认实现,这样上述接口就会变成合法的函数接口了。

@FunctionalInterface
public interface FunInterfaceAnnotationDemo {

        public void absMethodDemo();
        default public void absMoreMethodDemo(){
            System.out.println("Default implementation of method ininterface");
        }

一旦有了default 关键字,现在的接口就好像变成了抽象类。这让人总觉得怪怪的。这个default的接口主要是为了向后兼容。当你往一个拥有众多子类的接口中添加方法时,你总不想一个子类一个子类的去把方法实现吧?这在Java JDK不断升级的过程中,是一个很现实的问题。至于说使用带default实现的接口还是抽象类,我认为在进行Java代码的设计时,还是抽象类更合适。



郑重声明:本站内容如果来自互联网及其他传播媒体,其版权均属原媒体及文章作者所有。转载目的在于传递更多信息及用于网络分享,并不代表本站赞同其观点和对其真实性负责,也不构成任何其他建议。