Java8-Stream流式计算
什么是Stream流式计算
大数据:存储 + 计算
集合、MySQL 本质就是存储东西的;
计算都应该交给流来操作!
Stream是对集合功能的增强,它提供了各种非常便利、高效的聚合操作,可以大批量数据操作,同时再结合Lambda表达式,就可以极大的提高编程效率。
Stream的API提供了串行和并行两种模式进行操作数据:
串行处理操作:
- Stream 在执行每一步中间操作时,并不会做实际的数据操作处理,而是将这些中间操作串联起来,生成一个数据处理链表,通过 Spliterator 迭代器进行数据处理。
- 每执行一次迭代,就对所有的无状态的中间操作进行数据处理,而对有状态的中间操作,就需要迭代处理完所有的数据,再进行处理操作
- 最后进行终结操作的数据处理。
并行处理操作(parallel()):
- Stream 对中间操作基本跟串行处理方式是一样的
- 终结操作中,Stream 将结合ForkJoin对集合进行切片处理,ForkJoin 框架将每个切片的处理结果 Join 合并起来
Stream操作分为中间操作或者最终操作两种:
- 中间操作,返回Stream本身,这样就可以将多个操作依次串起来例如,
map、flatMap、filter、distinct、sorted、peek、limit、skip、parallel、sequential、unordered
- 最终操作,返回一特定类型的计算结果例如,
forEach、forEachOrdered、toArray、reduce、collect、min、max、count、anyMatch、allMatch、noneMatch、findFirst、findAny、iterator
Stream流举例
public class User {
private Integer id;
private String name;
private Integer age;
// get、set
// 有参、无参构造方法
// tostring
}
package com.kuang.stream;
import java.util.Arrays; import java.util.List;
/**
* 题目要求:一分钟内完成此题,只能用一行代码实现! * 现在有5个用户!筛选:
* 1、ID 必须是偶数
* 2、年龄必须大于23岁
* 3、用户名转为大写字母
* 4、用户名字母倒着排序
* 5、只输出一个用户!
*/
public class Test {
public static void main(String[] args) {
User u1 = new User(1,"a",21);
User u2 = new User(2,"b",22);
User u3 = new User(3,"c",23);
User u4 = new User(4,"d",24);
User u5 = new User(6,"e",25);
// 集合就是存储
List<User> list = Arrays.asList(u1, u2, u3, u4, u5);
// 计算交给Stream流
// lambda表达式、链式编程、函数式接口、Stream流式计算
list.stream()
.filter(u->{return u.getId()%2==0;})
.filter(u->{return u.getAge()>23;})
.map(u->{return u.getName().toUpperCase();})
// 这个只是字符串比较 如果是对象比较需要另一种方式
.sorted((uu1,uu2)->{return uu2.compareTo(uu1);})
.limit(1)
.forEach(System.out::println);
}
}
Stream流常用方法
iterator
:返回迭代器对象forEach
:将调Stream中的每个元素,交给一个Consumer函数处理
public class Test {
public static void main(String[] args{
Stream<String> stream = Stream.of("hello","world","briup");
stream.forEach(System.out::println);
}
}
count
:统计流中的元素数,并返回结果
public class Test {
public static void main(String[] args{
Stream<String> stream = Stream.of("hello","world","briup");
System.out.println(stream.count());
}
}
max
:返回流中基于comparator所指定的比较规则,比较出的最大值
Optional<T> max(Comparator<? super T> comparator);
toArray
:使用调用流中的元素,生成数组返回filter
:过滤
public static void filter(){
// 通过流将字符串转换成Stream格式的集合
Stream<String> java = Stream.of("Java", "Python", "PHP");
// 将包含字符串P的字符串筛选出来,并打印结果
java.filter(str -> str.contains("P")).forEach(str -> System.out.println(str));
}
distinct
:去重
System.out.println("所有user的年龄集合");
List<Integer> userAgeList = list.stream().map(User::getAge).distinct().collect(Collectors.toList());
System.out.println("userAgeList = " + userAgeList);
// ===========================================================
// 所有user的年龄集合
// userAgeList = [``10``, ``12``, ``15``, ``25``, ``16``, ``14``, ``17``]
Collectors.toList()
:返回一个 Collector ,它将输入元素 List到一个新的 Listlimit
:limit返回包含前n个元素的流,当集合大小小于n时,则返回实际长度
public static void filter(){
// 通过流将字符串转换成Stream格式的集合
Stream<String> java = Stream.of("Java", "Python", "PHP");
// 将包含字符串P的字符串筛选出来,只输出一个
java.filter(str -> str.contains("P")).limit(1).forEach(str -> System.out.println(str));
}
sorted
:sorted要求待比较的元素必须实现comparator接口
Comparator接口里面有一个compare方法,方法有两个参数T o1和T o2,是泛型的表示方式,分别表示待比较的两个对象
1、o1大于o2,如果返回正整数,表示参数o1大于o2
2、o1等于o2,返回0/3、o1小于o2,如果返回负整数,表示参数o1小于o2
import java.util.Comparator;
public class StudentComparator implements Comparator<StudentBean>{
public int compare(StudentBean o1, StudentBean o2) {
// 返回负数 说明比较器认为传进来得参数o1比o2小,所以o1在o2前面
// 实际上从我们角度o1比o2大
// 但是比较器只看返回得结果去判断o1是否大于o2从而进行升序降序
if(o1.getScore() > o2.getScore()){
return -1;
}else if(o1.getScore() < o2.getScore()){
return 1;
}
return 0;
}
}
map
:映射
假设我们希望筛选出所有专业为计算机科学的学生姓名,那么我们可以在filter筛选的基础之上,通过map将学生实体映射成为学生姓名字符串
List<String> names = students.stream()
.filter(student -> "计算机科学".equals(student.getMajor()))
.map(Student::getName).collect(Collectors.toList());
除了上面这类基础的map,java8还提供了mapToDouble(ToDoubleFunction mapper),mapToInt(ToIntFunction mapper),mapToLong(ToLongFunction mapper),这些映射分别返回对应类型的流,java8为这些流设定了一些特殊的操作,比如我们希望计算所有专业为计算机科学学生的年龄之和,那么我们可以实现如下:
int totalAge = students.stream()
.filter(student -> "计算机科学".equals(student.getMajor()))
.mapToInt(Student::getAge).sum();
-
调用串行Stream的parallel()方法,可以将其转换并行Stream
-
flatMap
:flatMap与map的区别在于 flatMap是将一个流中的每个值都转成一个个流,然后再将这些流扁平化成为一个流 。
举例说明,假设我们有一个字符串数组
String[] strs = {"java8", "is", "easy", "to", "use"};
我们希望输出构成这一数组的所有非重复字符,那么我们可能首先会想到如下实现:
List<String[]> distinctStrs = Arrays.stream(strs)
// 映射成为Stream<String[]>
.map(str -> str.split(""))
.distinct()
.collect(Collectors.toList());
在执行map操作以后,我们得到是一个包含多个字符串(构成一个字符串的字符数组)的流,此时执行
distinct操作是基于在这些字符串数组之间的对比,所以达不到我们希望的目的,此时的输出为:
[j, a, v, a, 8]
[i, s]
[e, a, s, y]
[t, o]
[u, s, e]
方法 | 说明 |
---|---|
filter | 元素过滤,对Stream对象按照指定的Predicate进行过滤,返回的Strema对象中仅包含满足条件的元素 |
map [mapToInt] [mapToLong] [mapToDouble] | 元素一对一转换,使用传入的Function对象对Stream中所有元素进行映射处理,返回的Stream对象中的元素为原元素处理后的结果 |
flatMap (flatMapToInt) (flatMapToLong) (flatMapToDouble) | 元素一对多转换,对Stream对象中的所有元素进行操作,每个元素会有一个或多个结果,然后将所有的元素组合成一个统一的Stream并返回 |
distinct | 元素去重,返回去重后的Stream对象 |
sorted [sorted(Comparator<? super T> comparator)] | 元素排序,返回排序后的Stream对象 |
limit | 元素截取,返回有限个元素组成新的Stream对象 |
skip | 元素跳过,抛弃前指定个元素后,使用剩下的元素组成新的Stream对象返回 |
peek | 生成一个包含原Stream的所有元素的新Stream,同时会提供一个消费函数即引用的方法,当Stream每个元素被消费的时候都会先执行新Stream给定的方法 |
distinct只有对于一个包含多个字符的流进行操作才能达到我们的目的,即对Stream<String>
进行操作。
此时flatMap就可以达到我们的目的:
List<String> distinctStrs = Arrays.stream(strs)
// 映射成为Stream<String[]>
.map(str -> str.split(""))
// 扁平化为Stream<String>
.flatMap(Arrays::stream)
.distinct()
.collect(Collectors.toList());
flatMap将由map映射得到的Stream<String[]>
,转换成由各个字符串数组映射成的流Stream<String>
,再将这些小的流扁平化成为一个由所有字符串构成的大流Steam<String>
,从而能够达到我们的目的。