Stream 流
简介
Java 8 中,得益于 lambda 带来的函数式编程,引入了一个全新的 Stream流 概念,用于解决集合已有的弊端。
好处
我们先来看一个例子:
筛选出 names 中以 '张' 开头的字符串得到子集1,再筛选出 子集1 中长度为 3 的字符串,然后遍历输出。
// 传统集合的方式
public static void main(String[] args) {
List<String> names = new ArrayList<>();
names.add("张三丰");
names.add("张大宝");
names.add("张三");
names.add("德玛杰");
names.add("乔峰");
names.add("李大宝");
List<String> list1 = new ArrayList<>(); //以 '张' 开头的字符串
for (String name:names){
if (name.startsWith("张")){
list1.add(name);
}
}
List<String> list2 = new ArrayList<>(); // list1中长度为 3 的字符串
for (String name:list1){
if (name.length()==3){
list2.add(name);
}
}
for (String name:list2){
System.out.println(name);
}
}
// stream流方式
public static void main(String[] args) {
List<String> names = new ArrayList<>();
names.add("张三丰");
names.add("张大宝");
names.add("张三");
names.add("德玛杰");
names.add("乔峰");
names.add("李大宝");
names.stream().filter(name->name.startsWith("张"))
.filter(name->name.length()==3).forEach(System.out::println);
}
可以看到用 stream 流方式比传统集合的方式精简了很多。
获取流的两种方式
- 所有的
Collection集合可以通过stream()方法获取流. Stream类的静态方法of()通常用于将数组转成stream流。
public static void main(String[] args) {
// 集合转成 stream 流
List<String> list = new ArrayList<>();
Stream<String> listStream = list.stream();
Set<Integer> set = new HashSet<>();
Stream<Integer> setStream = set.stream();
// 数组转成 stream 流
String[] names = new String[]{"zs","ls","ww"};
Stream<String> namesStream = Stream.of(names);
Stream<Integer> integerStream = Stream.of(1, 2, 3, 4, 5);
}
方法分类
stream 流中方法分为两类:
- 延迟方法,返回值仍是
Stream类型 - 终结方法,返回值不再是
Stream类型,例如:count()和forEach()。
常用方法--fliter()
filter() 将一个流转换成另外一个子流,该方法接受一个 Predicate 类型的参数(系统内置函数接口之一)。
Stream<T> filter(Predicate<? super T> predicate);
public static void main(String[] args) {
List<String> names = new ArrayList<>();
names.add("张三丰");
names.add("张大宝");
names.add("张三");
names.add("德玛杰");
names.add("乔峰");
names.add("李大宝");
// 过滤 names 集合,只要姓张的人,然后输出
names.stream().filter(name->name.startsWith("张")).
forEach(System.out::println);
}
Stream流特点--只能使用一次
Stream 流属于管道流,只能被使用一次,当一个 Stream 流调用了终结方法后,就会被关闭,再次调用其方法会报错。
public static void main(String[] args) {
// 数组转成 stream 流
String[] names = new String[]{"zsf","ls","ww"};
Stream<String> namesStream = Stream.of(names);
namesStream.filter(name->name.length()==3).
forEach(System.out::println);
// 流已经被关闭,再次调用
namesStream.forEach(System.out::println);
}

常用方法--map()
如果需要将流中的元素映射到另一个流中,可以用 map() 方法,一般用于将一种类型的数据转成另一种类型的数据。
<R> Stream<R> map(Function<? super T, ? extends R> mapper);
public static void main(String[] args) {
// 数组转成 stream 流
String[] strings = new String[]{"1","2","3"};
Stream<String> stream1 = Stream.of(strings);
// 使用 map 方法,将字符串类型的整数转成 Integer 类型的整数,然后遍历
Stream<Integer> stream2 = stream1.map(k->Integer.parseInt(k));
stream2.forEach(System.out::println);
}
常用方法--limit()
limit() 可以对流中的元素进行截取,只截取前 n 个
public static void main(String[] args) {
// 数组转成 stream 流
String[] strings = new String[]{"1","2","3","4"};
// 只截取前3个元素,输出
Stream.of(strings).limit(3).forEach(System.out::println);
}
常用方法--skip()
跳过流中的前 n 个元素,如果 n 大于流中元素的个数,返回一个长度为 0 的流。
public static void main(String[] args) {
// 数组转成 stream 流
String[] strings = new String[]{"1","2","3","4"};
// 跳过前2个元素,输出
Stream.of(strings).skip(2).forEach(System.out::println);
}
常用方法--concat()
Stream 类的静态方法用于将两个流合并。
public static void main(String[] args) {
// 数组转成 stream 流
String[] strings1 = new String[]{"1","2","3","4"};
String[] strings2 = new String[]{"3","4","5","6"};
Stream<String> stream1 = Stream.of(strings1);
Stream<String> stream2 = Stream.of(strings2);
Stream<String> stream3 = Stream.concat(stream1,stream2);
stream3.forEach(System.out::println); // 输出 1,2,3,4,3,4,5,6
}