序
写Java代码的程序员,集合的遍历是常有的事,用惯了for循环、while循环、do while循环,我们来点别的,JDK8 使用了新的forEach机制,结合streams,让你的代码看上去更加简洁、更加高端,便于后续的维护和阅读。好,不说了,"talk is cheap, show me the code",我们直接上代码,秉承一贯以来的风格。skr~skr~
一、对常用集合的遍历
JDK8中的forEach不仅可以对集合元素进行遍历,也能根据集合元素的值搞些事情,比如做判断,比如过滤。我们拿常用的List和Map来举例。
对Map集合的遍历:
/**
* 对Map的遍历
*/
Map<String, Integer> map = Maps.newHashMap();
map.put("天猫双11", 1024);
map.put("京东双11", 1024);
// ①简写式
map.forEach((k, v) -> System.out.println("key:" + k + ", value:" + v));
// ②判断式
map.forEach((k, v) -> {
System.out.println("key:" + k + ", value:" + v);
if (StringUtils.contains(k, "京东")) {
System.out.println("skr~");
}
});
复制代码
执行结果:
key:京东双11, value:1024
key:天猫双11, value:1024
key:京东双11, value:1024
skr~
key:天猫双11, value:1024
复制代码
对List集合的遍历:
/**
* 对List的遍历
*/
List<String> list = Lists.newArrayList();
list.add("买买买");
list.add("剁剁剁");
// ①简写式
list.forEach(item -> System.out.println(item));
// ②判断式
list.forEach(item -> {
if (StringUtils.contains(item, "买")) {
System.out.println("不如再用两个肾换个iPhone XS Max Pro Plus也无妨啊~");
}
});
复制代码
执行结果如下:
买买买
剁剁剁
不如再用两个肾换个iPhone XS Max Pro Plus也无妨啊~
复制代码
二、JDK8 中双冒号的使用
JDK8中有双冒号的用法,就是把方法当做参数传到stream内部,使stream的每个元素都传入到该方法里面执行一下。
比如,上面的List<String>
的打印,我们可以这样写:
// JDK8 双冒号的用法
list.forEach(System.out::println);
复制代码
执行结果也是一样一样的:
买买买
剁剁剁
复制代码
在 JDK8 中,接口Iterable 8默认实现了forEach方法,调用了 JDK8 中增加的接口Consumer内的accept方法,执行传入的方法参数。 JDK源码如下:
/**
* Performs the given action for each element of the {@code Iterable}
* until all elements have been processed or the action throws an
* exception. Unless otherwise specified by the implementing class,
* actions are performed in the order of iteration (if an iteration order
* is specified). Exceptions thrown by the action are relayed to the
* caller.
*
* @implSpec
* <p>The default implementation behaves as if:
* <pre>{@code
* for (T t : this)
* action.accept(t);
* }</pre>
*
* @param action The action to be performed for each element
* @throws NullPointerException if the specified action is null
* @since 1.8
*/
default void forEach(Consumer<? super T> action) {
Objects.requireNonNull(action);
for (T t : this) {
action.accept(t);
}
}
复制代码
三、对自定义类型的组装
这个用法我觉得是比较实用也比较常用的。我们先定义两个POJO,一个叫Track,是一个Entity,和我们的数据库表结构进行映射;另一个叫TrackVo,是一个Vo,在接口层返回给前端展示用。这里为了简化代码量,我们使用了lombok插件。好,先将它们定义出来:
Track.java
/**
* @author huangzx
* @date 2018/11/13
*/
@AllArgsConstructor
@Data
@Builder
public class Track {
private Long id;
private String name;
private String anchor;
}
复制代码
TrackVo.java
/**
* @author huangzx
* @date 2018/11/13
*/
@AllArgsConstructor
@Data
@Builder
public class TrackVo {
private Long trackId;
private String trackName;
}
复制代码
经常遇到的场景就是:我通过一个Dao层将数据fetch出来,是一个List<Track>
,但前端需要的是List<TrackVo>
,但Track和TrackVo的字段又不一样。按照之前的做法,可能是直接用一个for循环或while循环将List<Track>
遍历把里面的Entity赋值到TrackVo,你飞快地敲击键盘,伴随着屏幕的震动,十来行代码顿时被创造了出来,长舒一口气,大功告成!
殊不知,JDK8 自从引入新的forEach,结合streams,可以让这十来行代码浓缩为一行,实在是简练。来瞧一瞧:
/**
* 对自定义类型的组装
*/
List<Track> trackList = Lists.newArrayList();
Track trackFirst = Track.builder().id(1L).name("我的理想").anchor("方清平").build();
Track trackSecond = Track.builder().id(2L).name("台上台下").anchor("方清平").build();
trackList.add(trackFirst);
trackList.add(trackSecond);
List<TrackVo> trackVoList = trackList.parallelStream().map(track -> TrackVo.builder().trackId(track.getId()).trackName(track.getName()).build()).collect(Collectors.toList());
System.out.println(JSON.toJSONString(trackVoList));
复制代码
执行结果如下:
[{"trackId":1,"trackName":"我的理想"},{"trackId":2,"trackName":"台上台下"}]
复制代码
似不似和你预期的结果一样?
四、原理
ok,秉承程序员认识一件事物——“知其然必知其所以然”的原则。我们来分析一下forEach的实现原理。
首先,我们要了解一下上面用到的流 (streams) 概念,以及被动迭代器。
Java 8 最主要的新特性就是 Lambda 表达式以及与此相关的特性,如流 (streams)、方法引用 (method references)、功能接口 (functional interfaces)。正是因为这些新特性,我们能够使用被动迭代器而不是传统的主动迭代器,特别是 Iterable 接口提供了一个被动迭代器的缺省方法叫做 forEach()。缺省方法是 Java 8 的又一个新特性,是一个接口方法的缺省实现,在这种情况下,forEach() 方法实际上是用类似于 Java 5 这样的主动迭代器方式来实现的。
实现了 Iterable 接口的集合类 (如:所有列表 List、集合 map) 现在都有一个 forEach() 方法,这个方法接收一个功能接口参数,实际上传递给 forEach() 方法的参数是一个 Lambda 表达式。我们来编写一段使用 streams 的代码:
/**
* @author huangzx
* @date 2018/11/13
*/
public class StreamCountsTest {
public static void main(String[] args) {
List<String> words = Arrays.asList("natural", "flow", "of", "water", "narrower");
long count = words.stream().filter(w -> w.length() > 5).count();
System.out.println(count);
}
}
复制代码
上面所示代码使用 Java 8 方式编写代码实现流管道计算,统计字符长度超过5的单词的个数。列表 words 用于创建流,然后使用过滤器对数据集进行过滤,filter() 方法只过滤出单词的字符长度,该方法的参数是一个 Lambda 表达式。最后,流的 count() 方法作为最终操作,得到应用结果。
我们再对自定义类型的组装
那句代码作个解析,如下:
中间操作除了 filter() 之外,还有 distinct()、sorted()、map() 等,一般是对数据集的整理,返回值一般也是数据集。我们可以大致浏览一下它有哪些方法,如下:
总的来说,Stream 遵循 "做什么,而不是怎么去做" 的原则。在我们的示例中,描述了需要做什么,比如获得长单词并对它们的个数进行统计。我们没有指定按照什么顺序,或者在哪个线程中做。相反,循环在一开始就需要指定如何进行计算。
五、为什么要用它?
网上许多人说:JDK8 的 Lambda 表达式的性能不如传统书写方式的性能。那为何还要出现呢?JDK的新api和新语法有时并不是为了性能而去做极致优化的。从理论上来说,面向对象编程,性能相对面向过程肯定是降低的,但是可维护性或清晰度有了很大的提升。
所以一个特性用与不用,取决于你关注什么,当公司给你3个月时间去做功能实现的时候,显然不会有人花1个月去做性能优化,这时候更清晰合理的代码就很重要了,大多数时候性能问题不是来自于算法和api的平庸表现,而是出自各种系统的bug。
六、总结
总结一下上面讲了什么?首先是对于常见集合我们怎么用forEach去操作,并且介绍了双冒号的用法;然后基于一个已存在的集合怎么利用它产生一个新的集合,以封装成我们想要的数据;最后介绍了它的实现原理并阐释为何要用它的的原因。好了,下课。。。(老师~再见~~)