前言:
在实际开发中经常需要获取各种各样不同格式的数据,因为数据库的表结构是一开始就设计好的所以很多时候我们不得不先从数据库里或其他地方获得数据后再根据需求去一层一层的筛选数据,在Java 8之前的做法不外乎就是各种List、Set一起上,各种循环判断。如果只是简单的需求还好说,循环个一两次再判断一下就可以解决,但是需求复杂的话就会写出很复杂的代码出来,时间久了后不仅自己看不懂,而且因为代码太复杂测试的时候不敢保证覆盖到所有的地方,万一代码到了线上出问题,不仅不好改还可能影响到以前的功能,而在Java 8中就提供了新的特性stream,专门用于对集合中的数据进行筛选、分类等操作,下面我们就会介绍stream的使用。
stream使用:
我们来看下面这段代码,Dish里面有一个属性calories
代表菜品的卡路里值,现在的需求是按卡路里对菜品进行排序再返回菜名,并且要求卡路里的值大于400。我们可以看到下面的操作非常繁琐,先筛选出卡路里大于400的放入集合中,然后对这个集合进行排序,最后循环这个集合把名字放入一个新的集合,这是一个很常见的代码,在Java 8之前都是这么写的,但是这还只是对一个属性的筛选、排序就这么繁琐了,如果是多个属性会更加麻烦,下面我们就看看用stream是怎么实现的。
//将卡路里低于400的对象都放到集合中
List<Dish> lowCaloricDishes = new ArrayList<>();
for(Dish d: menu){
if(d.getCalories() < 400){
lowCaloricDishes.add(d);
}
}
//把这个集合排序
Collections.sort(lowCaloricDishes, new Comparator<Dish>() {
public int compare(Dish d1, Dish d2){
return Integer.compare(d1.getCalories(), d2.getCalories());
}
});
//再新建一个String类型的集合来存放菜名
List<String> lowCaloricDishesName = new ArrayList<>();
for(Dish d: lowCaloricDishes){
lowCaloricDishesName.add(d.getName());
}
这是Java 8中stream的用法,我们会发现这段代码写起来非常舒服,首先调用stream()
方法获取了集合menu
的流,然后调用了filter
方法来筛选出卡路里超过400的元素,接着调用了sorted
方法对筛选出来的元素进行排序,再调用map
方法把筛选出来的元素里面的name
属性抽出来作为一个新的流,最后一步则是调用collect
方法把存放name
的流转为List
格式返回,得到的结果和上面一模一样,但是整个步骤一目了然,先做什么后做什么,非常清晰,这就是stream的用法。
import static java.util.Comparator.comparing;
import static java.util.stream.Collectors.toList;
List<Dish> menu = ...
List<String> lowCaloricDishesName = menu.stream()
//筛选出卡路里大于400的
.filter(d -> d.getCalories() < 400)
//按卡路里值排序
.sorted(comparing(Dish::getCalories))
//抽取名字属性创建一个新的流
.map(Dish::getName)
//这个流按List类型返回
.collect(toList());
stream的定义:
讲到这里大家可能会产生疑惑,流到底是什么,为什么它可以进行这样的操作。简单来说流其实就是 “从支持数据处理操作的源生成的元素序列” ,这句话到底是什么意思呢?下面我们来看看流的各种特性就明白了。
- 元素序列,可以通过流访问到特定类型的一组有序的值,就像我们通过集合访问这些数据一样,只不过集合是为了存储数据,而流则注重与对数据的计算和处理。
- 源,流可以会从一个数据源那里获取数据并生成流,例如上面讲的从集合中获取,生成的流里面的元素和数据源里的一致。
- 数据处理操作,就像第一条里说的,流是注重与对数据的计算和处理,所以它有很多不同的方法可以对数据进行操作,例如筛选、分类、排序等等。
- 流水线,顾名思义,流的很多操作都会返回一个新的流,所以我们上面可以使用连缀的方式调用那些方法,其实每个操作都是基于它前面的那个操作返回的流进行的。
我们来看下面的这段代码和它的流程图。它一开始从menu
这个集合中获取到了流,流里面的数据和集合是相同的,然后调用filter
方法之后筛选出符合条件的元素组成一个新的流传递给了map
方法,然后map
方法又从这个流里面把元素的名字取出来组成一个新流传递给limit
方法,limit
方法则只取了前3个组成一个流传递给collect
方法然后返回一个List
,这就是流的工作原理
List<String> list = menu.stream()
//筛选出卡路里高于300的元素
.filter(d -> d.getCalories() > 300)
//获取名字组成的流
.map(Dish::getName)
//只取前3个
.limit(3)
//返回List格式
.collect(toList());
System.out.println(list );
stream只能遍历一次:
流和集合不一样,集合可以想遍历多少次就遍历多少次,但是流只能遍历一次,如果你做了下面这样的操作,那代码会抛出一个java.lang.IllegalStateException
异常,流已被操作或关闭。
List<String> title = Arrays.asList("Java8", "In", "Action");
Stream<String> s = title.stream();
s.forEach(System.out::println);
s.forEach(System.out::println);
stream的两种操作:
还是继续看这段代码,其中filter
、map
和limit
操作被称为 中间操作 , 中间操作 会返回一个新的流,而collect
则被称为 终端操作 ,只有终端操作才会让整个流执行并关闭,也就是说只有当collect
被执行时filter
、map
和limit
才会被执行,上面讲到流只能遍历一次就是因为forEach
就是一个终端操作,执行完后就关闭流了。
List<String> list = menu.stream()
//筛选出卡路里高于300的元素
.filter(d -> d.getCalories() > 300)
//获取名字组成的流
.map(Dish::getName)
//只取前3个
.limit(3)
//返回List格式
.collect(toList());
以上就是stream的基本使用方法了,当然它的功能还远远不止这些,后面我们还会讲到筛选、分组、查找等更强大的操作。
**粗体** _斜体_ [链接](http://example.com) `代码` - 列表 > 引用
。你还可以使用@
来通知其他用户。