Java 8 引入了函数式接口(Functional Interfaces)的概念:如果一个接口只定义了唯一一个抽象方法,那么这个接口就成为函数式接口。同时,Java 8 中还引入了一个新的注解:@FunctionalInterface。可以把这个注解放在一个接口前,用于表示这个接口是一个函数式接口。这个注解是非必须的,只要接口只定义了唯一一个抽象方法,虚拟机就会自动判断出该接口为一个函数式接口,不过最好在定义的函数式接口上加上 @FunctionalInterface 注解。对于加了 @FunctionalInterface 注解的接口,Java 编译器会在编译时对接口进行检查,如果发现该接口中定义的抽象方法不是一个,那么编译器会报错,这样就可以保证该接口的定义符合预期。
java.lang.Runnable 就是一个典型的函数式接口:
@FunctionalInterface
public interface Runnable {
void run();
}
另外,为了避免开发者创建大量的仅为 Lambda 表达式服务的函数式接口,Java 8 在 java.util.function 中增加了不少新的函数式通用接口。例如:
Function<T,R> :将 T 作为输入,返回 R 作为输出,他还包含了和其他函数组合的默认方法。
Predicate<T> :将 T 作为输入,返回一个布尔值作为输出,该接口包含多种默认方法来将 Predicate 组合成其他复杂的逻辑(与、或、非)。
Consumer<T> :将 T 作为输入,不返回任何内容,表示在单个参数上的操作。
简单讲解一下 Function<T,R>接口,它的定义如下:
@FunctionalInterface
public interface Function {
R apply(T var1);
default Function compose(Functionsuper V, ? extends T> var1) {
Objects.requireNonNull(var1);
return (var2) -> {
return this.apply(var1.apply(var2));
};
}
default Function andThen(Functionsuper R, ? extends V> var1) {
Objects.requireNonNull(var1);
return (var2) -> {
return var1.apply(this.apply(var2));
};
}
static Function identity() {
return (var0) -> {
return var0;
};
}
}
可以看到 Function 接口中只有一个抽象方法 R apply(T var1) 这个方法唯一一个待实现的方法(抽象方法),这个方法将 Function 对象应用到输入的参数上,然后返回计算结果。
default Function compose(Functionsuper V, ? extends T> var1) 为一个默认方法,如不需要重写可以保留其默认实现。它返回一个先执行传入的函数对象的 apply 方法再执行当前函数对象的 apply 方法的函数对象。
default Function andThen(Functionsuper R, ? extends V> var1) 方法与它的上一个方法很相似,它返回一个先执行当前函数对象的 apply 方法再执行传入的函数对象的 apply 方法的函数对象。
static Function identity() 为一个静态方法,它返回一个执行了 apply()方法之后只会返回输入参数(即 apply()方法中直接输出输入的数据)的函数对象。
欢迎来到这里!
我们正在构建一个小众社区,大家在这里相互信任,以平等 • 自由 • 奔放的价值观进行分享交流。最终,希望大家能够找到与自己志同道合的伙伴,共同成长。
注册 关于