1.什么是函数式接口(Functional Interface)?
你肯定见过这样的代码:
new Thread(()->System.out.println("funtional")).start();
非常的简洁,设计者为了让现有的函数基础上友好的支持Lambda,。最终采取的方法是:增加函数式接口的概念。
所谓的函数式接口,当然首先是一个接口,然后就是在这个接口里面只能有一个抽象方法。
这种类型的接口也称为SAM接口,即Single Abstract Method interfaces
。
那么问题来了,为什么需要它,它存在的意义是什么呢?
它表达了 Lambda 表达式的类型,函数式接口是方法签名(signature),lambda表达式是方法body,两者组成了一个整体。
比如java.lang.Runnable
与java.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.函数式接口里允许定义的方法
- 函数式接口里是可以包含默认方法,因为默认方法不是抽象方法,其有一个默认实现,所以是符合函数式接口的定义的;
- 函数式接口里是可以包含静态方法,因为静态方法不能是抽象方法,是一个已经实现了的方法,所以是符合函数式接口的定义的;
函数式接口里是可以包含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
中定义的接口,都非常有趣:
- Consumer – 在T上执行一个操作,无返回结果
- Supplier –无输入参数,返回T的实例
- Predicate –输入参数为T的实例,返回boolean值
- Function –输入参数为T的实例,返回R的实例