建议将所有使用迭代器这种数据处理模式处理集合的代码都转换成Stream API
的方式。Stream API
能更清晰地表达数据处理管道的意图。除此之外,通过短路和延迟载入以及利用现代计算机的多核架构,我们可以对Stream
进行优化。
比如,下面的命令式代码使用了两种模式:筛选和抽取,这两种模式被混在了一起,这样的代码结构迫使程序员必须彻底搞清楚程序的每个细节才能理解代码的功能。此外,实现需要并行运行的程序所面对的困难也多得多:
List<String> dishNames = new ArrayList<>();
for(Dish dish: menu) {
if(dish.getcalories() > 300) {
dishNames.add(dish.getName());
}
}
替代方案使用Stream API
,采用这种方式编写的代码读起来更像是问题陈述,并行化也非常容易:
menu.parallelStream()
.filter(d -> d.getcalories() > 300)
.map(Dish::getName)
.collect(toList());
将命令式的代码结构转换为Stream API
的形式是个困难的任务,因为需要考虑控制流语句,比如break
、continue
、return
,并选择使用恰当的流操作。
Lambda
表达式有利于行为参数化。可以使用不同的Lambda
表示不同的行为,并将它们作为参数传递给函数去处理执行。这种方式可以从容地面对需求的变化。比如,可以用多种方式为Predicate
创建筛选条件,或者使用Comparator
对多种对象进行比较。现在,来看看哪些模式可以马上应用到现有的代码中,享受Lambda
表达式带来的便利。
首先,没有函数接口,就无法使用Lambda
表达式。因此,需要在代码中引入函数接口。在什么情况下使用它们呢?这里介绍两种通用的模式,可以依照这两种模式重构代码,利用Lambda
表达式带来的灵活性,它们分别是:有条件的延迟执行和环绕执行。
经常看到这样的代码,控制语句被混杂在业务逻辑代码之中。典型的情况包括进行安全性检查以及日志输出。比如,下面的这段代码,它使用了Java
语言内置的Logger
类:
if (logger.isLoggable(Log.FINER)) {
logger.finer("Problem:" + generateDiagnostic());
}
这段代码的问题不少:
isLoggable
方法暴露给了客户端代码。更好的方案是使用log
方法,该方法在输出日志消息之前,会在内部检查日志对象是否已经设置为恰当的日志等级:
logger.log(Leve1.FINER, "Problem:" + generateDiagnostic());
这种方式更好的原因是不再需要在代码中插入那些条件判断,与此同时日志器的状态也不再被暴露出去。不过,这段代码依旧存在一个问题。日志消息的输出与否每次都需要判断,即使你已经传递了参数,不开启日志。
这就是Lambda
表达式可以施展拳脚的地方。你需要做的仅仅是延迟消息构造,如此一来,日志就只会在某些特定的情况下才开启(以此为例,当日志器的级别设置为FINER
时)。显然,Java 8
的API
设计者们已经意识到这个问题,并由此引入了一个对log
方法的重载版本,这个版本的log
方法接受一个Supplier
作为参数。这个替代版本的log
方法的函数签名如下:
public void log(Level level, Supplier<String> msgSupplier)
可以通过下面的方式对它进行调用:
logger.log(Level.FINER, () -> "Problem:" + generateDiagnostic());
如果日志器的级别设置恰当,log
方法会在内部执行作为参数传递进来的Lambda
表达式。这里介绍的Log
方法的内部实现如下:
public void log(Level level, Supplier<String> msgSupplier) {
if(logger.isLoggable(level)) {
log(level, msgSupplier.get()); //执行Lambda表达式
}
}
如果你发现你需要频繁地从客户端代码去查询一个对象的状态(比如前文例子中的日志器的状态),只是为了传递参数、调用该对象的一个方法(比如输出一条日志),那么可以考虑实现一个新的方法,以Lambda
或者方法表达式作为参数,新方法在检查完该对象的状态之后才调用原来的方法。你的代码会因此而变得更易读(结构更清晰),封装性更好(对象的状态也不会暴露给客户端代码了)。
如果发现虽然你的业务代码千差万别,但是它们拥有同样的准备和清理阶段,这时,完全可以将这部分代码用Lambda
实现。这种方式的好处是可以重用准备和清理阶段的逻辑,减少重复冗余的代码。下面这段代码,在打开和关闭文件时使用了同样的逻辑,但在处理文件时可以使用不同的Lambda
进行参数化。
String oneLine = processFile((BufferedReader b) -> b.readLine()); //传入一个Lambda表达式
String twoLines = processFile((BufferedReader b) -> b.readLine() + b.readLine()); //传入另一个Lambda表达式
public static String processFile(BufferedReaderProcessor p) throws IOException {
try(BufferedReader br = new BufferedReader(new FileReader("javasinaction/chap8/data.txt"))) {
return p.process(br); //将BufferedReaderProcessor作为执行参数传入
public interface BufferedReaderProcessor { 使用Lambda表达式的函数接口,该接口能够抛出一个IOException
String process(BufferedReader b) throws IOException;
}
这一优化是凭借函数式接口BufferedReaderProcessor
达成的,通过这个接口,可以传递各种Lambada
表达式对BufferedReader
对象进行处理。