《JAVA8实战》读书笔记之传递方法和传递lambda

传递方法:

假设 你有一个Apple类,它 有一个getColor方法,还有一个变量inventory保存着一个Apples的列表。你可能想要选出所 有的绿苹果,并返回一个列表。通常我们用筛选(filter)一词来表达这个概念。在Java 8之前, 你可能会写这样一个方法filterGreenApples:

 1 public static List<Apple> filterGreenApples( List<Apple> inventory )
 2 {
 3     List<Apple> result = new ArrayList<>(); for ( Apple apple : inventory )
 4     {
 5         if ( "green".equals( apple.getColor() ) )
 6         {
 7             result.add( apple );
 8         }
 9     }
10     return(result);
11 }

 

但是接下来,有人可能想要选出重的苹果,比如超过150克: 

 public static List<Apple> filterHeavyApples( List<Apple> inventory )
{
     List<Apple> result = new ArrayList<>();
     for ( Apple apple : inventory )
     {
        if ( apple.getWeight() > 150 )
        {
             result.add( apple );
         }
     }
     return(result);
 }

这 两个方法只有一行不同: if里面高亮的那行条件。

Java 8你可以这样写:

 1 public static boolean isGreenApple( Apple apple )
 2 {
 3     return("green".equals( apple.getColor() ) );
 4 }
 5 
 6 
 7 public static boolean isHeavyApple( Apple apple )
 8 {
 9     return(apple.getWeight() > 150);
10 }
11 
12 
13 public interface Predicate<T>{ 
14     boolean test( T t ); 
15 } 
16 
17 static List<Apple> filterApples( List<Apple> inventory, Predicate<Apple> p )
18 {
19     List<Apple> result = new ArrayList<>();
20     for ( Apple apple : inventory )
21     {
22         if ( p.test( apple ) )
23         {
24             result.add( apple );
25         }
26     }
27     return(result);
28 }        

要用它的话,你可以写: 

1 filterApples(inventory, Apple::isGreenApple);
2 或者 
3 filterApples(inventory, Apple::isHeavyApple);

 

注:

什么是谓词? 前 面 的 代 码 传 递 了 方 法 Apple::isGreenApple ( 它 接 受 参 数 Apple 并 返 回 一 个 boolean)给filterApples,后者则希望接受一个Predicate<Apple>参数。谓词(predicate)它接受一个参数值,并返回true或false。你 在后面会看到, Java 8也会允许你写Function<Apple,Boolean>,但用Predicate<Apple>是更标准的方式,效率也会更高一 点儿,这避免了把boolean封装在Boolean里面

 

传递 Lambda

把方法作为值来传递显然很有用,但要是为类似于isHeavyApple和isGreenApple这种可能只用一两次的短方法写一堆定义有点儿烦人。不过Java 8也解决了这个问题,它引入了一套新
记法(匿名函数或Lambda),让你可以写

filterApples(inventory, (Apple a) -> "green".equals(a.getColor()) );

或者

filterApples(inventory, (Apple a) -> a.getWeight() > 150 );

甚至

filterApples(inventory, (Apple a) -> a.getWeight() < 80 || "brown".equals(a.getColor()) );


所以,你甚至都不需要为只用一次的方法写定义;代码更干净、更清晰,因为你用不着去找自己到底传递了什么代码。但要是Lambda的长度多于几行(它的行为也不是一目了然)的话,那你还是应该用方法引用来指向一个有描述性名称的方法,而不是使用匿名的Lambda。你应该以代码的清晰度为准绳。

posted on 2018-01-19 23:15  近博  阅读(5504)  评论(1编辑  收藏  举报

导航