Java8:函数式接口(Functional Interface)

1.什么是函数式接口(Functional Interface)?

你肯定见过这样的代码:

new Thread(()->System.out.println("funtional")).start();

非常的简洁,设计者为了让现有的函数基础上友好的支持Lambda,。最终采取的方法是:增加函数式接口的概念。

所谓的函数式接口,当然首先是一个接口,然后就是在这个接口里面只能有一个抽象方法。
这种类型的接口也称为SAM接口,即Single Abstract Method interfaces

那么问题来了,为什么需要它,它存在的意义是什么呢?
它表达了 Lambda 表达式的类型,函数式接口是方法签名(signature),lambda表达式是方法body,两者组成了一个整体。
比如java.lang.Runnablejava.util.concurrent.Callable是函数式接口最典型的两个例。

2.自定义函数式接口及使用

定义了一个函数式接口如下:

@FunctionalInterface
interface Inter1{
    void printStr();
}

那么就可以使用Lambda表达式来表示该接口的一个实现(注:JAVA 8 之前一般是用匿名类实现的):

String str = "hello";
Inter1 inter1 = () -> System.out.println(str+":lc");
inter1.printStr();

3.注意:

@FunctionalInterface注解
Java 8为函数式接口引入了一个新注解@FunctionalInterface,主要用于编译级错误检查,加上该注解,当你写的接口不符合函数式接口定义的时候,编译器会报错。

接口中申明的方法的参数列表以及返回值要与Lambda表达式中的对应。

4.函数式接口里允许定义的方法

  1. 函数式接口里是可以包含默认方法,因为默认方法不是抽象方法,其有一个默认实现,所以是符合函数式接口的定义的;
  2. 函数式接口里是可以包含静态方法,因为静态方法不能是抽象方法,是一个已经实现了的方法,所以是符合函数式接口的定义的;
  3. 函数式接口里是可以包含Object里的public方法,这些方法对于函数式接口来说,不被当成是抽象方法(虽然它们是抽象方法);因为任何一个函数式接口的实现,默认都继承了Object类,包含了来自java.lang.Object里对这些抽象方法的实现;

    @FunctionalInterface
    interface Inter1{
        void printStr();
        default void  sayHello() {
            System.out.println("hello");
        }
        static void sysFuck() {
            System.out.println("fuck");
        }
        @Override
        boolean equals(Object obj);
    }
    

5.java.util.function

Lambda表达式在运行期表示为一个函数接口(functionalinterface)(或者说一个SAM类型),函数接口是一种只定义了一个抽象方法的接口。尽管JDK已经有一些接口都符合函数接口定义,比如Runnable 和 Comparator,但是这对API演进来说是显然不够的。我们又不能到处在代码里使用像Runnable这样的接口,因为这么做不合乎逻辑。
JDK8中新增了一个包java.util.function,这个包里有一些专门给新增的API使用的函数接口。

下面列出几个java.util.function中定义的接口,都非常有趣:

  1. Consumer – 在T上执行一个操作,无返回结果
  2. Supplier –无输入参数,返回T的实例
  3. Predicate –输入参数为T的实例,返回boolean值
  4. Function –输入参数为T的实例,返回R的实例

猜你喜欢

转载自blog.csdn.net/TheLudlows/article/details/78782844