Function类包含四种方法,其中一个抽象方法apply(),两个default方法andThen()和compose(),以及一个静态方法identity()。
实例化Function的时候需要实现其中的apply()方法,apply方法接收一个模板类型作为输入参数,在andThen()和compose()方法里会自动调用apply()方法。
andThen方法接收一个Function类的实例,通过andThen可以将任意多个Function的apply方法调用连接起来,在Java8里的源码如下:
default <V> Function<T, V> andThen(Function<? super R, ? extends V> after) {
Objects.requireNonNull(after);
return (T t) -> after.apply(apply(t));
测试用例:
Function<String, String> function1 = string -> {
System.out.println("function1输出了: " + string);
return string;
};
Function<String, String> function2 = string -> {
System.out.println("function2输出了: " + string);
return string;
};
function1.andThen(function2).apply("hello world");
输出结果
function1输出了: hello world
function2输出了: hello world
可以看到调用顺序是先调用funtion1然后调用function2。
compose方法和andThen方法一样接收一个另一个Function作为参数,但是顺序与andThen恰恰相反。
在Java8里的源代码如下:
default <V> Function<V, R> compose(Function<? super V, ? extends T> before) {
Objects.requireNonNull(before);
return (V v) -> apply(before.apply(v));
}
接下来是测试用例,保持前面的不变,只把最后一句由andThen改成
function1.compose(function2).apply("hello world");
输出结果:
function2输出了: hello world
function1输出了: hello world
可以看到这次先执行了function2之后再执行function1。
identity方法是一个静态方法,作用是返回一个Function对象,返回的对象总是返回它被传入的值。
源代码很简单
static <T> Function<T, T> identity() {
return t -> t;
}
总结:
util里的function包里并不仅仅只有这四个类,只是其中绝大部分都是由这四种衍生而来的,这个包主要是用于实现Java8最大的特性函数式编程,所以在很多其他的包的一些类中的很多地方都用到了这个function包里的类,更多的情况是作为方法中的一个匿名对象参数来使用,配合简洁的lambda表达式使得程序可读性变得更好。