[JAVA] Java8新特性 StreamAPI实例详解

1863 0
王子 2022-11-8 17:16:17 | 显示全部楼层 |阅读模式
目录

    Stream结果收集
      结果收集到集合中结果收集到数组中对流中的数据做聚合计算对流中数据做分组操作对流中的数据做分区操作对流中的数据做拼接
    并行的Stream流
      串行的Stream流并行流
        获取并行流并行流操作
      并行流和串行流对比线程安全问题



Stream结果收集

面试官:说说你常用的StreamAPI。

结果收集到集合中
  1.     public static void main(String[] args){
  2.         // Stream<String> stream = Stream.of("aa", "bb", "cc");
  3.         List<String> list = Stream.of("aa", "bb", "cc","aa")
  4.                 .collect(Collectors.toList());
  5.         System.out.println(list);
  6.         // 收集到 Set集合中
  7.         Set<String> set = Stream.of("aa", "bb", "cc", "aa")
  8.                 .collect(Collectors.toSet());
  9.         System.out.println(set);
  10.         // 如果需要获取的类型为具体的实现,比如:ArrayList HashSet
  11.         ArrayList<String> arrayList = Stream.of("aa", "bb", "cc", "aa")
  12.                 //.collect(Collectors.toCollection(() -> new ArrayList<>()));
  13.                 .collect(Collectors.toCollection(ArrayList::new));
  14.         System.out.println(arrayList);
  15.         HashSet<String> hashSet = Stream.of("aa", "bb", "cc", "aa")
  16.                 .collect(Collectors.toCollection(HashSet::new));
  17.         System.out.println(hashSet);
  18.     }
复制代码
输出:
[aa, bb, cc, aa]
[aa, bb, cc]
[aa, bb, cc, aa]
[aa, bb, cc]

结果收集到数组中

Stream中提供了toArray方法来将结果放到一个数组中,返回值类型是Object[],如果我们要指定返回的类型,那么可以使用另一个重载的toArray(IntFunction f)方法。
  1.     public static void main(String[] args){
  2.         Object[] objects = Stream.of("aa", "bb", "cc", "aa")
  3.                 .toArray(); // 返回的数组中的元素是 Object类型
  4.         System.out.println(Arrays.toString(objects));
  5.         // 如果我们需要指定返回的数组中的元素类型
  6.         String[] strings = Stream.of("aa", "bb", "cc", "aa")
  7.                 .toArray(String[]::new);
  8.         System.out.println(Arrays.toString(strings));
  9.     }
复制代码
对流中的数据做聚合计算

当我们使用Stream流处理数据后,可以像数据库的聚合函数一样对某个字段进行操作,比如获得最大值,最小值,求和,平均值,统计数量。
  1.     public static void main(String[] args) {
  2.         // 获取年龄的最大值
  3.         Optional<Person> maxAge = Stream.of(
  4.                 new Person("张三", 18)
  5.                 , new Person("李四", 22)
  6.                 , new Person("张三", 13)
  7.                 , new Person("王五", 15)
  8.                 , new Person("张三", 19)
  9.         ).collect(Collectors.maxBy((p1, p2) -> p1.getAge() - p2.getAge()));
  10.         System.out.println("最大年龄:" + maxAge.get());
  11.         // 获取年龄的最小值
  12.         Optional<Person> minAge = Stream.of(
  13.                 new Person("张三", 18)
  14.                 , new Person("李四", 22)
  15.                 , new Person("张三", 13)
  16.                 , new Person("王五", 15)
  17.                 , new Person("张三", 19)
  18.         ).collect(Collectors.minBy((p1, p2) -> p1.getAge() - p2.getAge()));
  19.         System.out.println("最新年龄:" + minAge.get());
  20.         // 求所有人的年龄之和
  21.         Integer sumAge = Stream.of(
  22.                 new Person("张三", 18)
  23.                 , new Person("李四", 22)
  24.                 , new Person("张三", 13)
  25.                 , new Person("王五", 15)
  26.                 , new Person("张三", 19)
  27.         )
  28.                 //.collect(Collectors.summingInt(s -> s.getAge()))
  29.                 .collect(Collectors.summingInt(Person::getAge))
  30.                 ;
  31.         System.out.println("年龄总和:" + sumAge);
  32.         // 年龄的平均值
  33.         Double avgAge = Stream.of(
  34.                 new Person("张三", 18)
  35.                 , new Person("李四", 22)
  36.                 , new Person("张三", 13)
  37.                 , new Person("王五", 15)
  38.                 , new Person("张三", 19)
  39.         ).collect(Collectors.averagingInt(Person::getAge));
  40.         System.out.println("年龄的平均值:" + avgAge);
  41.         // 统计数量
  42.         Long count = Stream.of(
  43.                 new Person("张三", 18)
  44.                 , new Person("李四", 22)
  45.                 , new Person("张三", 13)
  46.                 , new Person("王五", 15)
  47.                 , new Person("张三", 19)
  48.         ).filter(p->p.getAge() > 18)
  49.                 .collect(Collectors.counting());
  50.         System.out.println("满足条件的记录数:" + count);
  51.     }
复制代码
对流中数据做分组操作

当我们使用Stream流处理数据后,可以根据某个属性将数据分组。
  1.     public static void main(String[] args){
  2.         // 根据账号对数据进行分组
  3.         Map<String, List<Person>> map1 = Stream.of(
  4.                 new Person("张三", 18, 175)
  5.                 , new Person("李四", 22, 177)
  6.                 , new Person("张三", 14, 165)
  7.                 , new Person("李四", 15, 166)
  8.                 , new Person("张三", 19, 182)
  9.         ).collect(Collectors.groupingBy(Person::getName));
  10.         map1.forEach((k,v)-> System.out.println("k=" + k +"\t"+ "v=" + v));
  11.         System.out.println("-----------");
  12.         // 根据年龄分组 如果大于等于18 成年否则未成年
  13.         Map<String, List<Person>> map2 = Stream.of(
  14.                 new Person("张三", 18, 175)
  15.                 , new Person("李四", 22, 177)
  16.                 , new Person("张三", 14, 165)
  17.                 , new Person("李四", 15, 166)
  18.                 , new Person("张三", 19, 182)
  19.         ).collect(Collectors.groupingBy(p -> p.getAge() >= 18 ? "成年" : "未成年"));
  20.         map2.forEach((k,v)-> System.out.println("k=" + k +"\t"+ "v=" + v));
  21.     }
复制代码
输出结果:
k=李四    v=[Person{name='李四', age=22, height=177}, Person{name='李四', age=15, height=166}]
k=张三    v=[Person{name='张三', age=18, height=175}, Person{name='张三', age=14, height=165}, Person{name='张三', age=19, height=182}]
-----------
k=未成年    v=[Person{name='张三', age=14, height=165}, Person{name='李四', age=15, height=166}]
k=成年    v=[Person{name='张三', age=18, height=175}, Person{name='李四', age=22, height=177}, Person{name='张三', age=19, height=182}]
多级分组: 先根据name分组然后根据年龄分组。
  1.     public static void main(String[] args){
  2.         // 先根据name分组,然后根据age(成年和未成年)分组
  3.         Map<String,Map<Object,List<Person>>> map =  Stream.of(
  4.                 new Person("张三", 18, 175)
  5.                 , new Person("李四", 22, 177)
  6.                 , new Person("张三", 14, 165)
  7.                 , new Person("李四", 15, 166)
  8.                 , new Person("张三", 19, 182)
  9.         ).collect(Collectors.groupingBy(
  10.                 Person::getName
  11.                 ,Collectors.groupingBy(p->p.getAge()>=18?"成年":"未成年"
  12.                 )
  13.         ));
  14.         map.forEach((k,v)->{
  15.             System.out.println(k);
  16.             v.forEach((k1,v1)->{
  17.                 System.out.println("\t"+k1 + "=" + v1);
  18.             });
  19.         });
  20.     }
复制代码
输出结果:
李四
    未成年=[Person{name='李四', age=15, height=166}]
    成年=[Person{name='李四', age=22, height=177}]
张三
    未成年=[Person{name='张三', age=14, height=165}]
    成年=[Person{name='张三', age=18, height=175}, Person{name='张三', age=19, height=182}]

对流中的数据做分区操作

Collectors.partitioningBy会根据值是否为true,把集合中的数据分割为两个列表,一个true列表,一个false列表。
  1.     public static void main(String[] args){
  2.         Map<Boolean, List<Person>> map = Stream.of(
  3.                 new Person("张三", 18, 175)
  4.                 , new Person("李四", 22, 177)
  5.                 , new Person("张三", 14, 165)
  6.                 , new Person("李四", 15, 166)
  7.                 , new Person("张三", 19, 182)
  8.         ).collect(Collectors.partitioningBy(p -> p.getAge() > 18));
  9.         map.forEach((k,v)-> System.out.println(k+"\t" + v));
  10.     }
复制代码
输出结果:
false    [Person{name='张三', age=18, height=175}, Person{name='张三', age=14, height=165}, Person{name='李四', age=15, height=166}]
true    [Person{name='李四', age=22, height=177}, Person{name='张三', age=19, height=182}]

对流中的数据做拼接

Collectors.joining会根据指定的连接符,将所有的元素连接成一个字符串。
  1.     public static void main(String[] args){
  2.         String s1 = Stream.of(
  3.                 new Person("张三", 18, 175)
  4.                 , new Person("李四", 22, 177)
  5.                 , new Person("张三", 14, 165)
  6.                 , new Person("李四", 15, 166)
  7.                 , new Person("张三", 19, 182)
  8.         ).map(Person::getName)
  9.                 .collect(Collectors.joining());
  10.         // 张三李四张三李四张三
  11.         System.out.println(s1);
  12.         String s2 = Stream.of(
  13.                 new Person("张三", 18, 175)
  14.                 , new Person("李四", 22, 177)
  15.                 , new Person("张三", 14, 165)
  16.                 , new Person("李四", 15, 166)
  17.                 , new Person("张三", 19, 182)
  18.         ).map(Person::getName)
  19.                 .collect(Collectors.joining("_"));
  20.         // 张三_李四_张三_李四_张三
  21.         System.out.println(s2);
  22.         String s3 = Stream.of(
  23.                 new Person("张三", 18, 175)
  24.                 , new Person("李四", 22, 177)
  25.                 , new Person("张三", 14, 165)
  26.                 , new Person("李四", 15, 166)
  27.                 , new Person("张三", 19, 182)
  28.         ).map(Person::getName)
  29.                 .collect(Collectors.joining("_", "###", "$$$"));
  30.         // ###张三_李四_张三_李四_张三$$$
  31.         System.out.println(s3);
  32.     }
复制代码
并行的Stream流


串行的Stream流

我们前面使用的Stream流都是串行,也就是在一个线程上面执行。

并行流

parallelStream其实就是一个并行执行的流,它通过默认的ForkJoinPool,可以提高多线程任务的速度。

获取并行流

我们可以通过两种方式来获取并行流。
    通过List接口中的parallelStream方法来获取通过已有的串行流转换为并行流(parallel)
  1.     public static void main(String[] args){
  2.         List<Integer> list = new ArrayList<>();
  3.         // 通过List 接口 直接获取并行流
  4.         Stream<Integer> integerStream = list.parallelStream();
  5.         // 将已有的串行流转换为并行流
  6.         Stream<Integer> parallel = Stream.of(1, 2, 3).parallel();
  7.     }
复制代码
并行流操作
  1.     public static void main(String[] args){
  2.         Stream.of(1,4,2,6,1,5,9)
  3.                 .parallel() // 将流转换为并发流,Stream处理的时候就会通过多线程处理
  4.                 .filter(s->{
  5.                     System.out.println(Thread.currentThread() + " s=" +s);
  6.                     return s > 2;
  7.                 }).count();
  8.     }
复制代码
并行流和串行流对比

我们通过for循环,串行Stream流,并行Stream流来对500000000亿个数字求和。来看消耗时间。
  1. public class Test {
  2.     private static long times = 500000000;
  3.     private  long start;
  4.     @Before
  5.     public void befor(){
  6.         start = System.currentTimeMillis();
  7.     }
  8.     @After
  9.     public void end(){
  10.         long end = System.currentTimeMillis();
  11.         System.out.println("消耗时间:" + (end - start));
  12.     }
  13.     /**
  14.      * 普通for循环 消耗时间:138
  15.      */
  16.     @Test
  17.     public void test01(){
  18.         System.out.println("普通for循环:");
  19.         long res = 0;
  20.         for (int i = 0; i < times; i++) {
  21.             res += i;
  22.         }
  23.     }
  24.     /**
  25.      * 串行流处理
  26.      *   消耗时间:203
  27.      */
  28.     @Test
  29.     public void test02(){
  30.         System.out.println("串行流:serialStream");
  31.         LongStream.rangeClosed(0,times)
  32.                 .reduce(0,Long::sum);
  33.     }
  34.     /**
  35.      * 并行流处理 消耗时间:84
  36.      */
  37.     @Test
  38.     public void test03(){
  39.         LongStream.rangeClosed(0,times)
  40.                 .parallel()
  41.                 .reduce(0,Long::sum);
  42.     }
  43. }
复制代码
通过案例我们可以看到parallelStream的效率是最高的。
Stream并行处理的过程会分而治之,也就是将一个大的任务切分成了多个小任务,这表示每个任务都是一个线程操作。

线程安全问题

在多线程的处理下,肯定会出现数据安全问题。如下:
  1.     @Test
  2.     public void test(){
  3.         List<Integer> list = new ArrayList<>();
  4.         for (int i = 0; i < 1000; i++) {
  5.             list.add(i);
  6.         }
  7.         System.out.println(list.size());
  8.         List<Integer> listNew = new ArrayList<>();
  9.         // 使用并行流来向集合中添加数据
  10.         list.parallelStream()
  11.                 //.forEach(s->listNew.add(s));
  12.                 .forEach(listNew::add);
  13.         System.out.println(listNew.size());// 839
  14.     }
复制代码
针对这个问题,我们的解决方案有哪些呢?
    加同步锁使用线程安全的容器通过Stream中的toArray/collect操作
以上就是Java8新特性 StreamAPI实例详解的详细内容,更多关于Java8新特性 StreamAPI的资料请关注中国红客联盟其它相关文章!
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

中国红客联盟公众号

联系站长QQ:5520533

admin@chnhonker.com
Copyright © 2001-2025 Discuz Team. Powered by Discuz! X3.5 ( 粤ICP备13060014号 )|天天打卡 本站已运行