What is Stream ?
data:image/s3,"s3://crabby-images/0e685/0e685d7312ff05f3386b0225597e2313fd7cdbbb" alt="在这里插入图片描述"
注意:
data:image/s3,"s3://crabby-images/0257b/0257b1d96ee537af0efd92d0027869fed2b75141" alt="在这里插入图片描述"
Stream操作三部曲
data:image/s3,"s3://crabby-images/a64d8/a64d868595c0496e4180485d782cca0f5b2bbf73" alt="在这里插入图片描述"
使用演示:
@Test
void test()
{
List<String> list=new ArrayList<>();
Stream<String> stream = list.stream();
People[] peoples=new People[10];
Stream<People> stream1 = Arrays.stream(peoples);
Stream<String> aa = Stream.of("aa", "bb", "cc");
Stream.iterate(0, (x) -> x + 2)
.limit(10)
.forEach(System.out::println);
}
}
data:image/s3,"s3://crabby-images/fafb1/fafb1ebcbbef2a81fe07f11abf98253bcf7a28e3" alt="在这里插入图片描述"
中间操作
data:image/s3,"s3://crabby-images/e19eb/e19eb95f207b7254324e34d25894e147dd647699" alt="在这里插入图片描述"
筛选与切片
filter---接收Lambda,从流中排除某些元素
limit(max)---截断流,使其元素不超过给定数量
skip(n)---跳过元素,返回一个扔掉了前n个的元素的流,若流张元素不足n个,则返回一个空流,与limit(n)互补
distinct---筛选,通过流所生成的元素的hashcode()和equals()去重复元素
data:image/s3,"s3://crabby-images/b7168/b71688feb78aa745640ff63326af1bc8802da1a5" alt="在这里插入图片描述"
内部迭代: 迭代操作由Stream API完成
终止操作:一次性执行全部内容,即惰性求值
使用演示:
public class TestMain
{
List<People> peopleList= Arrays.asList(
new People("1号",18,3000),
new People("2号",21,4000),
new People("3号",19,5000),
new People("4号",20,3500)
);
@Test
void test()
{
Stream<People> s=peopleList.stream().filter(people ->people.getAge()>19);
s.forEach(System.out::println);
}
}
data:image/s3,"s3://crabby-images/87c5a/87c5a05199f6cd76451953b06122a11a78f9c24d" alt="在这里插入图片描述"
外部迭代
Iterator<People> iterator = peopleList.iterator();
while(iterator.hasNext())
System.out.println(iterator.next());
limit ===> 短路
public class TestMain
{
List<People> peopleList= Arrays.asList(
new People("1号",18,3000),
new People("2号",21,4000),
new People("3号",19,5000),
new People("4号",20,3500)
);
@Test
void test()
{
peopleList.stream().
filter(people ->{
System.out.println("短路");
return people.getAge()>15;}).
limit(2).
forEach(System.out::println);
}
}
data:image/s3,"s3://crabby-images/0efde/0efdeda18587f23d0b8edef21ab9f62e0f3a847e" alt="在这里插入图片描述"
skip ===>跳过前n个元素
public class TestMain
{
List<People> peopleList= Arrays.asList(
new People("1号",18,3000),
new People("2号",21,4000),
new People("3号",19,5000),
new People("4号",20,3500)
);
@Test
void test()
{
peopleList.stream().
filter(people ->{
System.out.println("短路");
return people.getAge()>15;}).
skip(2).
forEach(System.out::println);
}
}
data:image/s3,"s3://crabby-images/74d05/74d0574358da0c9dd1de5d04b9df77c3e0402703" alt="在这里插入图片描述"
distinct进行元素去重(自定义类需要重写对应的hashcode和equals方法)
public class TestMain
{
List<People> peopleList= Arrays.asList(
new People("1号",18,3000),
new People("2号",21,4000),
new People("2号",21,4000),
new People("4号",20,3500)
);
@Test
void test()
{
peopleList.stream().
distinct().
forEach(System.out::println);
}
}
data:image/s3,"s3://crabby-images/3d695/3d695e57edc49983cafbabae28b49ad244d0d21d" alt="在这里插入图片描述"
映射
map–接收Lambda,将元素转换为其他形式或提取信息,接收一个函数作为参数,该函数会被应用到每个元素上,并将其映射成一个新的元素
flatMap—接收一个函数作为参数,将流中的每个值都换成另一个流,然后把所有流连接成一个流
map的使用演示:
public class TestMain
{
List<People> peopleList= Arrays.asList(
new People("1号",18,3000),
new People("2号",21,4000),
new People("2号",21,4000),
new People("4号",20,3500)
);
@Test
void test()
{
List<String> list=Arrays.asList("a","b","c");
list.stream().map((x)->x.toUpperCase()).forEach(System.out::println);
System.out.println(list);
System.out.println("------------------------------------------------");
peopleList.stream().map(People::getName).forEach(System.out::println);
}
}
data:image/s3,"s3://crabby-images/81542/81542598e65ddfe2af2fdd8ecb1ac6bcd432dd81" alt="在这里插入图片描述"
flatMap使用演示:
使用前,先看一下下面这个案例:
void test()
{
List<String> list=Arrays.asList("aaa","bbb","ccc");
Stream<Stream<Character>> sm = list.stream().map(TestMain::getAll);
sm.forEach(System.out::println);
}
public static Stream<Character> getAll(String str)
{
List<Character> list=new ArrayList<>();
for(Character ch:str.toCharArray())
{
list.add(ch);
}
return list.stream();
}
data:image/s3,"s3://crabby-images/9d629/9d62935776f1a14f25b83c9c464d7f34a3bc765b" alt="在这里插入图片描述" 显然这里我们将list集合对应的新流中每一个元素,都映射为了一个流,并返回,相当于现在的大流中有三个小流
下面我们需要遍历这些小流,取出里面的值
void test()
{
List<String> list=Arrays.asList("aaa","bbb","ccc");
Stream<Stream<Character>> sm = list.stream().map(TestMain::getAll);
sm.forEach(x-> x.forEach(System.out::println));
}
public static Stream<Character> getAll(String str)
{
List<Character> list=new ArrayList<>();
for(Character ch:str.toCharArray())
{
list.add(ch);
}
return list.stream();
}
data:image/s3,"s3://crabby-images/0c6a2/0c6a2090a0cb641f4b6282d2ac1ce2ac011e6fb0" alt="在这里插入图片描述"
显然上面写法比较复杂,下面给出简化写法
@Test
void test()
{
List<String> list=Arrays.asList("aaa","bbb","ccc");
Stream<Character> characterStream = list.stream()
.flatMap(TestMain::getAll);
characterStream.forEach(System.out::println);
}
public static Stream<Character> getAll(String str)
{
List<Character> list=new ArrayList<>();
for(Character ch:str.toCharArray())
{
list.add(ch);
}
return list.stream();
}
data:image/s3,"s3://crabby-images/eed71/eed7137b3e03e9b2e10ad1f52a5d2fcbee83b2d6" alt="在这里插入图片描述"
map与flatmap的区别
data:image/s3,"s3://crabby-images/0a295/0a295b1fb1fe6c2e3f5b587b258c16be3c075080" alt="在这里插入图片描述" map是将对应的每个小流放入当前大流中构成一个流 data:image/s3,"s3://crabby-images/52107/521079825a88654f949b9eb3d2d7e733d086fbfa" alt="在这里插入图片描述" flatmap取出集合中的每个元素放入当前的流中,相当于将每个小流里面的元素拿出来组合为一个大流
这里还可以参考add()和addAll()的关系:
List<String> list=Arrays.asList("aaa","bbb","ccc");
List list1=new ArrayList();
list1.add(list);
list1.addAll(list);
System.out.println(list1);
data:image/s3,"s3://crabby-images/bf86c/bf86cde8b863c48e50fd4dd39a16abe6e3c66eb2" alt="在这里插入图片描述"
排序
sorted()—自然排序(Comparable)
sorted(Comparator com)—定制排序(Comparator)
List<People> peopleList= Arrays.asList(
new People("1号",18,3000),
new People("2号",21,4000),
new People("2号",21,4000),
new People("4号",18,3500)
);
@Test
void test()
{
peopleList.stream().sorted((x,y)->{
if(x.getAge()==y.getAge())
return -x.money.compareTo(y.getMoney());
else
return x.getAge().compareTo(y.getAge());
}).forEach(System.out::println);
}
data:image/s3,"s3://crabby-images/7eb13/7eb1332f430749cd151d48ca19dc64c0c2a1eb1c" alt="在这里插入图片描述"
Stream的终止操作如下
data:image/s3,"s3://crabby-images/208be/208be53c03dc7a7bf00de38482b9eec6dfd77c58" alt="在这里插入图片描述"
查找与匹配
查找与匹配
allMatch--检查是否匹配所有元素
anyMatch---检查是否至少匹配一个元素
noneMatch---检查是否没有匹配所有元素
findFirst---返回第一个元素
findAny---返回当前流中任意元素
count---返回流中元素的总个数
max----返回流中最大值
min---返回流中最小值
演示:
public class TestMain
{
List<People> peopleList= Arrays.asList(
new People("1号",18,3000, People.STATUS.BUSY),
new People("2号",21,4000, People.STATUS.FREE),
new People("2号",21,4000, People.STATUS.BUSY),
new People("4号",18,3500, People.STATUS.BUSY)
);
@Test
void test()
{
boolean ret = peopleList.stream().allMatch(x -> x.getStatus().equals(People.STATUS.BUSY));
System.out.println(ret);
boolean ret1 = peopleList.stream().anyMatch(x -> x.getStatus().equals(People.STATUS.FREE));
System.out.println(ret1);
boolean ret2 = peopleList.stream().noneMatch(x -> x.getStatus().equals(People.STATUS.BUSY));
System.out.println(ret2);
Optional<People> first = peopleList.stream().sorted((x, y) -> -x.getMoney().compareTo(y.getMoney())).findFirst();
System.out.println(first.get());
Optional<People> any = peopleList.parallelStream().filter(x -> x.getStatus().equals(People.STATUS.FREE)).findAny();
System.out.println(any.get());
long count = peopleList.stream().count();
System.out.println(count);
Optional<People> max = peopleList.stream().max((x, y) -> -x.getMoney().compareTo(y.getMoney()));
System.out.println(max.get());
Optional<Integer> min = peopleList.stream().map(People::getMoney).min(Integer::compareTo);
System.out.println(min.get());
}
}
data:image/s3,"s3://crabby-images/83839/83839c55821dffd5c33a29457551febf9e20f41d" alt="在这里插入图片描述"
归约–reduce
public class TestMain
{
List<People> peopleList= Arrays.asList(
new People("1号",18,3000, People.STATUS.BUSY),
new People("2号",21,4000, People.STATUS.FREE),
new People("2号",21,4000, People.STATUS.BUSY),
new People("4号",18,3500, People.STATUS.BUSY)
);
@Test
void test()
{
Integer sum = peopleList.stream().map(People::getMoney).reduce(0, (x, y) -> x + y);
System.out.println("money总和为:"+sum);
}
}
data:image/s3,"s3://crabby-images/1790c/1790c82469291da617c243bd234221bec354ede9" alt="在这里插入图片描述" data:image/s3,"s3://crabby-images/5f40b/5f40b3780e89b796a8ed94c15b5ad32ab5dd477b" alt="在这里插入图片描述" 也可以不指定起始值,但是这样可能数据为空,因此会被封装为一个Optional对象
public class TestMain
{
List<People> peopleList= Arrays.asList(
new People("1号",18,3000, People.STATUS.BUSY),
new People("2号",21,4000, People.STATUS.FREE),
new People("2号",21,4000, People.STATUS.BUSY),
new People("4号",18,3500, People.STATUS.BUSY)
);
@Test
void test()
{
Optional<Integer> reduce = peopleList.stream().map(People::getMoney).reduce((x, y) -> x + y);
System.out.println("money总和为:"+reduce.get());
}
}
data:image/s3,"s3://crabby-images/687d0/687d0e4374b96798c0d0c9fb2fffbebaa03c55fe" alt="在这里插入图片描述"
这里不一定非要是数的累加,也可以是字符串的反复拼接
.reduce("",String::contact);
data:image/s3,"s3://crabby-images/d982b/d982b74f50295e649eafd351cd9ed02ca5cdea47" alt="在这里插入图片描述"
收集
collect----将流转换为其他形式,接收一个Collector接口的实现,用于给Stream中元素做汇总的方法
data:image/s3,"s3://crabby-images/3095e/3095efc1b7baae11be203efcdf154cde286fe4ce" alt="在这里插入图片描述" 演示:
public class TestMain
{
List<People> peopleList= Arrays.asList(
new People("aaa",18,3000, People.STATUS.BUSY),
new People("bbb",21,4000, People.STATUS.FREE),
new People("ccc",21,4000, People.STATUS.BUSY),
new People("ddd",18,3500, People.STATUS.BUSY)
);
@Test
void test()
{
Map<String, String> collect = peopleList.stream().map(People::getName)
.collect(Collectors.toMap(x -> x.toUpperCase(), y -> y));
System.out.println(collect);
System.out.println("==============================");
List<String> stringList = peopleList.stream().map(People::getName).collect(Collectors.toList());
System.out.println(stringList);
System.out.println("==============================");
HashSet<String> collect1 = peopleList.stream().
map(People::getName).collect(Collectors.toCollection(HashSet::new));
System.out.println(collect1);
}
}
data:image/s3,"s3://crabby-images/4e186/4e1862bee9facf33a8c469d3b12c01104e56a338" alt="在这里插入图片描述"
collect的其他一些用法
public class TestMain
{
List<People> peopleList= Arrays.asList(
new People("aaa",18,3000, People.STATUS.BUSY),
new People("bbb",21,4000, People.STATUS.FREE),
new People("ccc",21,4000, People.STATUS.BUSY),
new People("ddd",18,3500, People.STATUS.BUSY)
);
@Test
void test()
{
Long sum = peopleList.stream().collect(Collectors.counting());
System.out.println("当前流中元素的总数:"+sum);
Double MoneyAvg = peopleList.stream().collect(Collectors.averagingInt(People::getMoney));
System.out.println("工资平均值:"+MoneyAvg);
IntSummaryStatistics age = peopleList.stream().collect(Collectors.summarizingInt(People::getAge));
System.out.println("年龄所有相关的信息:"+age);
Integer ageSUm = peopleList.stream().collect(Collectors.summingInt(People::getAge));
System.out.println(ageSUm);
Optional<Integer> moneyMax = peopleList.stream().map(People::getMoney).collect(Collectors.maxBy((x,y)->Integer.compare(x,y)));
System.out.println("最高工资:"+moneyMax.get());
Optional<Integer> moneyMin = peopleList.stream().map(People::getMoney).collect(Collectors.minBy(Integer::compare));
System.out.println("最低工资:"+moneyMin.get());
}
}
data:image/s3,"s3://crabby-images/18570/18570ce80e9638ceeb5ba456d213e9a80f61a5fd" alt="在这里插入图片描述"
collect里面的分组
单级分组:
public class TestMain
{
List<People> peopleList= Arrays.asList(
new People("aaa",18,3000, People.STATUS.BUSY),
new People("bbb",21,4000, People.STATUS.FREE),
new People("ccc",21,4000, People.STATUS.FREE),
new People("ddd",18,3500, People.STATUS.BUSY)
);
@Test
void test()
{
Map<People.STATUS, List<People>> collect = peopleList.stream().collect(Collectors.groupingBy(People::getStatus));
System.out.println(collect);
}
}
data:image/s3,"s3://crabby-images/3384b/3384b872b3ef0ae4a4a12674e9e07a0ade1ee51c" alt="在这里插入图片描述" 多级分组:
public class TestMain
{
List<People> peopleList= Arrays.asList(
new People("aaa",18,3000, People.STATUS.BUSY),
new People("bbb",21,4000, People.STATUS.FREE),
new People("ccc",21,10000, People.STATUS.FREE),
new People("ddd",18,12000, People.STATUS.BUSY)
);
@Test
void test()
{
Map<People.STATUS, Map<String, List<People>>> collect = peopleList.stream().collect(Collectors.groupingBy(People::getStatus, Collectors.groupingBy(
x -> {
if (x.getMoney() >= 10000)
return "有钱人";
else
return "穷人";
}
)));
System.out.println(collect);
}
}
data:image/s3,"s3://crabby-images/269da/269da447002786c30c3aea8a1b3ce24967be0126" alt="在这里插入图片描述"
collect里面的分区
public class TestMain
{
List<People> peopleList= Arrays.asList(
new People("aaa",18,3000, People.STATUS.BUSY),
new People("bbb",21,4000, People.STATUS.FREE),
new People("ccc",21,10000, People.STATUS.FREE),
new People("ddd",18,12000, People.STATUS.BUSY)
);
@Test
void test()
{
Map<Boolean, List<People>> ret = peopleList.stream().collect(Collectors.partitioningBy(x -> x.getMoney() >= 10000));
System.out.println(ret);
}
}
data:image/s3,"s3://crabby-images/9d573/9d5730eec36f1a96d8d5a4da41b5910721c97bc2" alt="在这里插入图片描述"
collect里面获取某个属性相关的详细信息(平均值,最大值…)
public class TestMain
{
List<People> peopleList= Arrays.asList(
new People("aaa",18,3000, People.STATUS.BUSY),
new People("bbb",21,4000, People.STATUS.FREE),
new People("ccc",21,10000, People.STATUS.FREE),
new People("ddd",18,12000, People.STATUS.BUSY)
);
@Test
void test()
{
IntSummaryStatistics collect = peopleList.stream().collect(Collectors.summarizingInt(People::getMoney));
System.out.println(collect);
System.out.println(collect.getMax());
System.out.println(collect.getCount());
}
}
data:image/s3,"s3://crabby-images/5afc9/5afc94f8b33223e434f033fdd7480d9e82c03061" alt="在这里插入图片描述"
collect里面的join,完成字符串连接工作
public class TestMain
{
List<People> peopleList= Arrays.asList(
new People("aaa",18,3000, People.STATUS.BUSY),
new People("bbb",21,4000, People.STATUS.FREE),
new People("ccc",21,10000, People.STATUS.FREE),
new People("ddd",18,12000, People.STATUS.BUSY)
);
@Test
void test()
{
String ret = peopleList.stream().map(People::getName).collect(Collectors.joining(",", "==", "=="));
System.out.println(ret);
}
}
data:image/s3,"s3://crabby-images/1925d/1925ddf589ca8835459184672806d7191d388def" alt="在这里插入图片描述"
并行流与串行流
一、什么是并行流
并行流 : 就是把一个内容分成多个数据块,并用不同的线程分 别处理每个数据块的流。
Java 8 中将并行进行了优化,我们可以很容易的对数据进行并 行操作。Stream API 可以声明性地通过 parallel() 与 sequential() 在并行流与顺序流之间进行切换。
二、了解 Fork/Join框架
Fork/Join 框架 : 就是在必要的情况下,将一个大任务,进行拆分(fork)成若干个 小任务(拆到不可再拆时),再将一个个的小任务运算的结果进行 join 汇总.
data:image/s3,"s3://crabby-images/6da79/6da793188e124e0c7190f15f03e65bb319f774bf" alt="在这里插入图片描述"
三、Fork/Join 框架与传统线程池的区别
采用 “工作窃取”模式(work-stealing): 当执行新的任务时它可以将其拆分分成更小的任务执行,并将小任务加到线 程队列中,然后再从一个随机线程的队列中偷一个并把它放在自己的队列中。
相对于一般的线程池实现,fork/join框架的优势体现在对其中包含的任务的 处理方式上.在一般的线程池中,如果一个线程正在执行的任务由于某些原因 无法继续运行,那么该线程会处于等待状态。而在fork/join框架实现中,如果 某个子问题由于等待另外一个子问题的完成而无法继续运行.那么处理该子 问题的线程会主动寻找其他尚未运行的子问题来执行。这种方式减少了线程 的等待时间, 高了性能。
四、 案例
创建一个ForkJoinCalculate计算类:
public class ForkJoinCalculate extends RecursiveTask<Long> {
private long start;
private long end;
private static final long THRESHOLD = 1000000;
public ForkJoinCalculate(long start, long end) {
this.start = start;
this.end = end;
}
@Override
protected Long compute() {
long length = end - start;
if (length <= THRESHOLD) {
long sum = 0;
for (long i = start; i <= end; i++) {
sum += i;
}
return sum;
}else {
long middle = (start + end) / 2;
ForkJoinCalculate left = new ForkJoinCalculate(start, middle);
left.fork();
ForkJoinCalculate right = new ForkJoinCalculate(middle + 1, end);
right.fork();
return left.join() + right.join();
}
}
}
测试方法:
private static final long END_VALUE = 10000000000L;
@Test
public void test1(){
Instant start = Instant.now();
ForkJoinPool pool = new ForkJoinPool();
ForkJoinTask<Long> task = new ForkJoinCalculate(0, END_VALUE);
Long sum = pool.invoke(task);
System.out.println(sum);
Instant end = Instant.now();
System.out.println("耗时:" + Duration.between(start, end).toMillis());
}
执行结果:
-5340232216128654848
耗时:2325
使用普通for 循环:
@Test
public void test2(){
Instant start = Instant.now();
long sum = 0L;
for (long i = 0; i <= END_VALUE; i ++){
sum += i;
}
System.out.println(sum);
Instant end = Instant.now();
System.out.println("耗时:" + Duration.between(start, end).toMillis());
}
执行结果:
-5340232216128654848
耗时:3571
java8中 Fork/Join计算
@Test
public void test3(){
Instant start = Instant.now();
LongStream.rangeClosed(0, END_VALUE)
.parallel()
.reduce(0, Long::sum);
Instant end = Instant.now();
System.out.println("耗时为:" + Duration.between(start, end).toMillis());
}
执行结果:
耗时为:1690
检查本机的可用处理器数:
@Test
public void test4(){
int num = Runtime.getRuntime().availableProcessors();
System.out.println(num);
}
执行结果:
8
Optional类
data:image/s3,"s3://crabby-images/5ea81/5ea814b278729fccb8758c0d17a4518e417b38bd" alt="在这里插入图片描述"
Java 8 Optional的正确姿势
|