本节将重点探讨复合Lambda表达式的使用。在阅读本篇之前建议您先阅读:
java8实战读书笔记:Lambda表达式语法与函数式编程接口,因为本篇是上篇的补充。
本书为博主在学习《java8实战》时的笔记。
上面是JDK8中java.util.Comparator接口,相比jdk1.7增加了好多方法,也许你会觉得奇怪,为什么接口中还能定义方法,原因是JDK8中,可以为接口添加默认实现,使用default关键字定义。
我们可以这样定义一个比较器:
1Comparator< Apple> c = Comparator.comparing(Apple::getWeight());
其等价为:
1Comparator< Apple> c = Comparator.comparing( (a) -> a.getWeight() );
为什么可以这样写呢?因为Comparator定义了如下静态方法:
逆序
Comparator定义了一个静态方法,reversed,故我们不需要重新再定义一个比较器,我们可以这样就能实现逆序排序:
1List< Apple> apples = new ArrayList<>(); 2apples.sort( Comparator.comparing(Apple::getWeight()).reversed() );
比较器链
如果要支持多重排序呢?例如先根据苹果的重量,如果重量相同就按照颜色排序,那如何来实现呢?
1apples.sort( Comparator.comparing(Apple::getWeight()).thenComparing( Apple::getColor() ) );
之所以可以使用上述表达式,是因为Comparator定义了如下方法:
1default <U extends Comparable<? super U>> Comparator<T> thenComparing( Function<? super T, ? extends U> keyExtractor) 2{ 3 return thenComparing(comparing(keyExtractor)); 4}
温馨提示:Comparator中定义很多thenComparing重载方法,在具体使用过程中,可以先看看其函数声明。
提到谓词复合,我们就不得不提Predicate<T>函数式编程接口,其类图如下所示:
- and:与
- negate:非
- or:或
温馨提示:and 和 or 是按照在表达式链中的位置,从左向右确定优先级的。因此, a.or(b).and(c) 可以看作 (a || b) && c 。
使用示例:从苹果列表中找出所有红色的,并且重量超过150的苹果:
1apples.filter((a -> "red".equals(a.getColor())).and( a -> a.getWeight() > 150 )); 2a -> "red".equals(a.getColor()) 是 (Apple a ) -> "red".equals(a.getColor())的简写。
函数复合,其对应的函数式编程接口为Function,其类图如下:
- addThen
andThen 方法会返回一个函数,它先对输入应用一个给定函数,再对输出应用另一个函数。
例如:
Function f = x -> x + 1;
Function g = x -> x * 2;
Function h = f.andThen(g);
int result = h.apply(1); // 其结果返回4,类似与数学公式 f(g(x))。
- compose
先把给定的函数用作 compose 的参数里面给的那个函数,然后再把函数本身用于结果。与addThen的函数应用方向相反,同样举例说明如下:
Function f = x -> x + 1;
Function g = x -> x * 2;
Function h = f.compose(g);
int result = h.apply(1); // 其结果返回3,类似与数学公式 g(f(x))。