Java流式编程stream

news2024/11/19 23:40:06

文章目录

      • 一、简介
      • 二、创建Stream
      • 三、常用操作
      • 四、其他操作


image-20220809112034032

一、简介

  1. 流式 API 从 Java8 开始引入,支持链式书写。

  2. 流只能消费一次,不能被两次消费(两次最终操作)

  3. 流在管道中流通,在节点被处理。

  4. 流【无存储】,流不是一种数据结构,流存储的只是一种数据视图。

  5. Stream是独特的,既不同于io,也不同于List。

  6. Stream 不是 InputStream/OutputStream,它俩首先不在同一个包,其次概念上也不一样,Stream代表的是任意Java对象的序列。

    • java.util.stream

    • java.io

      image-20220809073626906

  7. Stream 不是 List,List中存储的元素是事先存在于内存中的Java对象,而Stream输出的元素可能并没有预先存储在内存中,而是通过实时计算出来的惰性对象。其次,Stream在理论上能容纳无限对象,List不能。

  8. 简单流式计算:

    cat.map(n -> n.multiply(n))
       .limit(100)
       .forEach(System.out::println);
    
  9. 流式计算特点:

    • 惰性计算,真正的计算通常只发生在最后的结果获取时。流式计算存在第一步、中间一步、最后一步的说法,只有当到达最后一步执行函数的时候,整个惰性函数才会执行。

    • 一个Stream可以轻易转换为另一个Stream。

    • 存在泛型接口,需要指明使用对象。

    • 流式计算应该和 Lambda 表达式一起使用,符合Java8的思想(两者同时升级出现)。

      cat.filter(n -> n % 2 == 0) // 不计算
         .map(n -> n * n) 				// 不计算
         .limit(100) 							// 不计算
         .sum(); 									// 计算
      
  10. 常用操作举例

中间操作最终操作
mapforEach
filtercollect
limitcount
sorted
distinct
  1. 【无限序列】说明

    ​ 对于无限序列,如果直接调用 forEach() 或者 count() 求最终值,会直接进入死循环,因为无限序列永远不可能被计算完。所以我们需要先将起转变为有序序列,例如limit(100)。

  2. 【并行计算】

    • 使用十分简单。

    • 正常情况下,Stream为单线程,现在我们想要多线程。

    • parallel(),自动转化多线程。

      stream.parallel()
      			.sorted() 
       			.toArray(String[]::new);
      

二、创建Stream

4种方法,

其中第1、2种方式创建出来的流顺序是固定的,3、4不固定。

  1. of():最简单的方法

    Stream<String> stream = Stream.of("a","b","c");
    stream.forEach(System.out::println);
    
  2. Collection:进阶

    // 方式一,Collection直接调用stream方法
    List list = List.of("a","b","c");
    Stream stream = list.stream();
    
    // 方式二,利用Arrays(本质和方式一差不多)
    Stream<String> s = Arrays.stream(new String[] { "A"});
    
  3. Supplier(“暗示”,存储算法):采用Stream.generate(),传入Supplier对象。基于Supplier创建的Stream会不断调用Supplier.get()方法来生成下一个元素,这种Stream中保留的不是元素,而是算法。范例:

    Stream<Integer> my = Stream.generate(new MySup());
    my.limit(10).forEach(System.out::println);
    
    // 不断生成自然数的Supplier(范围在Integer之内)
    class MySup implements Supplier<Integer> {
        int n = 0;
        public Integer get() {
            n++;
            return n;
        }
    }
    
  4. 其他API接口

    • Files类的lines()方法,常用于遍历文本文件。
    • 正则表达式Pattern对象存在 splitAsStream() 方法,可以直接把一个长字符串分割成Stream序列而不是数组。

三、常用操作

  1. 【三种】基本类型流

    ​ 在Java中,因为Java泛型不支持基本类型,所以我们无法使用像Stream<int>这样的形式来保存int,只能采用形如Integer这样的形式。但是频繁装箱、拆箱操作会牺牲编译器的大量性能。

    ​ 所以为了提高效率,Java标准库提供了三种使用基本类型的Stream,它们的使用和标准的Stream没有太大区别,直接使用:

    • IntStream
    • LongStream
    • DoubleStream
    // 1. 
    IntStream is = Arrays.stream(new int[] { 1, 2, 3 });
    
    // 2. 将Stream<String>转换为LongStream:
    LongStream s=List.of("1").stream().mapToLong(Long::parseLong);
    
  2. ==map()==方法简介

    • 映射操作,它将一个Stream转换为另一个Stream。
    • 每一次映射都会自赋值,形如:a = a + 1,所以不再需要编写复制语句(思想)。
    Stream<Integer> s1 = Stream.of(1,2,3);
    
    Stream s2 = s1.map(n->n*n);
    s2.forEach(System.out::println);
    
    • 如果我们查看Stream的源码,会发现map()方法接收的对象是Function接口对象,它定义了一个apply()方法,负责把一个T类型转换成R类型:
    <R> Stream<R> map(Function<? super T, ? extends R> mapper);
    
    @FunctionalInterface
    public interface Function<T, R> {
        // 将T类型转换为R:
        R apply(T t);
    }
    
    • map运算的综合应用案例
    List.of("  Apple ", " pear ")
        .stream() 												// 转换为Stream
        .map(String::trim) 								// 去空格
        .map(String::toLowerCase)					// 变小写
        .forEach(System.out::println); 		// 打印
    
  3. ==forEach()==遍历

    stream.forEach(System.out::println);
    
  4. filter():过滤,true则通过。

    IntStream.of(1, 2, 3, 4, 5, 6, 7, 8, 9)
      				// 注意lambda的编写形式
             .filter(n -> n % 2 != 0)
             .forEach(System.out::println);
    
  5. ==reduce()==方法简介

    • reduce()是聚合方法,map和filter是转换方法。

      • 转换方法:元素一一对应并转换。非“最终方法”,不会触发计算。
      • 聚合方法:合并计算所有值,得到一个最终结果。“最终方法”,会触发计算操作。
    • reduce()方法传入的对象是BinaryOperator接口实现,它定义了一个apply()方法,负责把上次累加的结果和本次的元素进行运算,并返回累加的结果。

    • 使用较为复杂,暂时跳过。

    @FunctionalInterface
    public interface BinaryOperator<T> {
        // Bi操作:两个输入,一个输出
        T apply(T t, T u);
    }
    
  6. ==collect()==方法简介:将Stream转化为集合。

    // 1. 过滤空白字符并转为 ArrayList
    stream.filter(s -> s != null && !s.isBlank())
      		.collect(Collectors.toList());
    
    // 2. 转为 Set
    stream.collect(Collectors.toSet());
    
    // 3. 转为 Map,先 key 后 value
    Stream<String> stream = Stream.of("APPL:Apple");
    
    Map<String, String> map = 
      stream.collect(Collectors.toMap(
      s -> s.substring(0, s.indexOf(':')),
      s -> s.substring(s.indexOf(':') + 1)));
    
  7. ==toArray()==方法简介:将Stream转化为数组。

    String[] array = list.stream().toArray(String[]::new);
    

四、其他操作

  1. collect分组输出

    • 实际.collect()方法。
    • 用途:导入学生名单,然后按班级分类输出。
    List<String> list = List.of("Apple", "Banana", "Blackberry", "Coconut", "Avocado", "Cherry", "Apricots");
    
    Map<String, List<String>> g = list.stream().collect(Collectors.groupingBy(s -> s.substring(0, 1), Collectors.toList()));
    
    System.out.println(groups);
    //输出:{A=[Apple, Avocado, Apricots], B=[Banana, Blackberry], C=[Coconut, Cherry]}
    
  2. sorted排序(两种方式)

    • 已经实现comparable接口的,直接使用;
    • 自定义Comparator比较器。

    【注意sordted只是转换操作,只会返回新的Stream而不会修改原来的Stream】

    // 无“自赋值”
    stream = stream.sorted(); 
    stream = stream.sorted(String::compareToIgnoreCase);
    
  3. distinct去重

    // 无“自赋值”
    stream = stream.distinct();
    
  4. limit截取:从头开始截取。

    // 无“自赋值”
    stream = stream.limit(20);
    
  5. skip: 扔掉前面几个元素。

    stream = stream.skip(5)
    
  6. concat合并流

    Stream<String> s1 = ... ;
    Stream<String> s2 = ... ;
    
    Stream<String> s3 = Stream.concat(s1,s2);
    
  7. flatMap平面化:将“三维数据”转化为“二维数据”。

    image-20220809103915928
    Stream< List<Integer> > s = Stream.of(
            Arrays.asList(1, 2, 3),
            Arrays.asList(4, 5, 6),
            Arrays.asList(7, 8, 9));
    
    Stream<Integer> i = s.flatMap(list -> list.stream());
    
  8. 数学计算

    • count():返回元素个数。
    • max(Comparator<? super T> cp):找出最大元素。
    • min(Comparator<? super T> cp):找出最小元素。
    • sum():对所有元素求和,只适用于数值类型。
    • average():对所有元素求平均数。
  9. 匹配

    • boolean allMatch(Predicate<? super T>):所有元素均满足测试条件。
    • boolean anyMatch(Predicate<? super T>):至少有一个元素满足测试条件。
  10. 复合实例

Stream<Student> ds= Stream.of(
new Student("zhang",99.6),
new Student("qian",78.12),
new Student("sun",59.9),
new Student("sun",60.0));

ds.filter(stu -> stu.getScore()<60.0)
  .forEach(System.out::println);

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/498.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

vim如何进行批量化注释及取消,也在1024表明自己算十分之一的程序员

前言 &#x1f47b;作者&#xff1a;龟龟不断向前 &#x1f47b;简介&#xff1a;宁愿做一只不停跑的慢乌龟&#xff0c;也不想当一只三分钟热度的兔子。 &#x1f47b;专栏&#xff1a;C初阶知识点 &#x1f47b;工具分享&#xff1a; 刷题&#xff1a; 牛客网 leetcode笔记软…

人家网站都免费了,你还用Python去爬?

文章目录⛳️ 实战场景⛳️ 实战编码⛳️ 实战场景 这次实战的目标是一个叫做猫肯的字体站点&#xff0c;该站点所有的字体都是免费可商用的&#xff0c;所以为什么还要去下载呢&#xff1f; 答案是练手&#xff0c;借免费站点学习爬虫&#xff0c;&#x1f30b; 目标站点地址…

Python爬虫技术系列-05字符验证码识别

Python爬虫技术系列-05字符验证码识别1. 光学文字识别1.1 OCR概述1.2 OCR识别库Tesseract下载安装1.3 生成验证码图片1.4 字符验证码识别1.安装python识别验证码库&#xff1a;2.验证码识别&#xff1a;1.5 使用打码平台识别验证码1.6 滑动验证码识别1. 光学文字识别 1.1 OCR概…

卡尔曼滤波实例——预测橘子的轨迹

目录 流程 一、采用轮廓的方式检测橘子位置 &#xff08;一&#xff09;滚动条获取阈值 &#xff08;二&#xff09;获取到图像中的包围橘子对应的白色图形的最小矩形框的信息 二、获取橘子检测框的质心 三、将质心送入卡尔曼滤波器&#xff0c;获取下一次的质心位置 四…

Markdown语言的简单学习

Markdown简单语法标题#空格 一级标题##空格 二级标题 以此类推三级标题四级...五级.....引用列表代码块表格分隔线链接强调语法&#xff08;斜体、加粗、下划线&#xff09;标题 #空格 一级标题 ##空格 二级标题 以此类推 三级标题 四级… 五级… … 引用 这是一段引用 …

<人生重开模拟器>——《Python项目实战》

目录 1.模拟实现 "人生重开模拟器" 1.1 问题导引&#xff1a; 1.2 问题分析&#xff1a; 2. 模拟实现分析及步骤&#xff1a; 3.完整源码&#xff1a; 4.写在最后的话&#xff1a; 后记&#xff1a;●由于作者水平有限&#xff0c;文章难免存在谬误之处&…

数据结构与算法----栈和队列(Stack Queue)

文章目录栈栈的操作栈的初始化入栈出栈取栈顶的元素判断栈是否为空求栈中数据元素的个数遍历栈中的所有元素清空栈栈的存储结构顺序存储链式存储顺序栈和链栈的区别栈的实战题目队列队列的操作入队出队遍历队列清空队列队列的存储结构顺序存储循环队列链式存储队列实战题目总结…

快速发布windows上的web项目【免费内网穿透】

快速发布windows上的web项目【免费内网穿透】 文章目录快速发布windows上的web项目【免费内网穿透】什么是cpolar内网穿透&#xff1f;概述1. 搭建一个静态Web站点1.1 下载演示站点1.2 本地运行演示站点1.3 本地浏览测试站点是否正常2. 注册并安装cpolar内网穿透3. 本地web站点…

玩转 CSS 的艺术之美

你将获得 深刻理解各种CSS原理 解构不为人知的CSS技巧 概念、技巧、场景三合一&#xff0c;实现“神奇”效果 强化吸收CSS知识体系&#xff0c;玩转各种神操作骚技巧 作者介绍 JowayYoung&#xff0c;资深前端工程师&#xff0c;目前就职于网易互动娱乐事业群&#xff0c…

前端面试之道

小册介绍 如果需要用一句话来介绍这本小册的话&#xff0c;「一年磨一剑」应该是最好的答案了。 为什么这样说呢&#xff1f;在出小册之前&#xff0c;我收集了大量的一线大厂面试题&#xff0c;通过大数据统计出了近百个常考知识点&#xff0c;然后根据这些知识点写成了这本…

《深度学习》:CANN训练营_昇腾AI入门课学习笔记(第二章 TensorFlow模型迁移训练)

文章目录第二章 TensorFlow模型迁移&训练本章学习目标AI模型开发基础知识入门Python水平要求了解深度学习和神经网络了解TensorFlow AI框架了解基于CANN的模型开发流程&#xff08;重点&#xff09;TensorFlow AI模型迁移详解为什么要做模型迁移TensorFlow AI模型自动迁移详…

【Python数据科学快速入门系列 | 10】Matplotlib数据分布图表应用总结

这是机器未来的第59篇文章 原文首发地址&#xff1a;https://robotsfutures.blog.csdn.net/article/details/127484292 《Python数据科学快速入门系列》快速导航&#xff1a; 【Python数据科学快速入门系列 | 01】Numpy初窥——基础概念【Python数据科学快速入门系列 | 02】创…

安卓讲课笔记3.3 相对布局

文章目录零、学习目标一、导入新课二、新课讲解&#xff08;一&#xff09;相对布局概述1、布局特点2、继承关系图3、常用属性&#xff08;1&#xff09;相对于父容器居中&#xff08;2&#xff09;相对于父容器对齐&#xff08;3&#xff09;相对于其它控件位置&#xff08;4&…

牛客网经典Java面试常见题

个人主页&#xff1a;熬夜磕代码丶 作品专栏: 数据结构与算法 我变秃了&#xff0c;也变强了 给大家介绍一款程序员必备刷题平台——牛客网 点击注册一起刷题收获大厂offer吧 文章目录一、二叉搜索树与双向链表二、从尾到头打印链表三、调整数组奇数位于偶数前面四、删除链表…

大数据毕业设计可视化大屏前后端项目分享

1、前言 很久没有分享过可视化大屏的项目了&#xff0c;距离上次分享基于Echarts的数据可视化大屏系统设计分享这篇可视化系统已经过去了整整一年有余。当时分享这篇博客没想到会收获这么多的阅读量&#xff0c;并且在刚发布的时候&#xff0c;还上了CSDN的博客热搜2&#xff…

gcc环境下演示C语言变长数组

前言 &#x1f47b;作者&#xff1a;龟龟不断向前 &#x1f47b;简介&#xff1a;宁愿做一只不停跑的慢乌龟&#xff0c;也不想当一只三分钟热度的兔子。 &#x1f47b;专栏&#xff1a;C初阶知识点 &#x1f47b;工具分享&#xff1a; 刷题&#xff1a; 牛客网 leetcode笔记软…

安全帽佩戴识别算法

安全帽佩戴识别算法采用SuiJi-AI人工智能深度学习技术计算机智能视觉识别算法&#xff0c;且通过规模化的安全帽数据识别训练。安全帽佩戴识别算法借助现场已有的监控摄像头对监控画面中人员着装行为进行实时分析识别。假如检测人员不戴安全帽&#xff0c;SuiJiAi将立即记录和警…

致敬第一个1024(第一次Java代码编程)

昨日&#xff0c;笔者刚刚下载好IDEA&#xff0c;今日&#xff0c;笔者将会实现第一次的Java代码&#xff01;&#xff01;信心满满&#xff01;&#xff01;下面请看笔者的代码吧&#xff01;&#xff01;骄傲的小心&#xff0c;已经澎湃&#xff01; 因此&#xff0c;本篇文…

如何实现一个SQL解析器

作者&#xff1a;vivo 互联网搜索团队- Deng Jie 一、背景 随着技术的不断的发展&#xff0c;在大数据领域出现了越来越多的技术框架。而为了降低大数据的学习成本和难度&#xff0c;越来越多的大数据技术和应用开始支持SQL进行数据查询。SQL作为一个学习成本很低的语言&#…

【正点原子I.MX6U-MINI应用篇】5、嵌入式Linux在LCD上显示BMP、JPG、PNG图片

一、BMP图像介绍与显示 我们常用的图片格式有很多&#xff0c;一般最常用的有三种&#xff1a;JPEG(或 JPG)、PNG、BMP和GIF。其中 JPEG(或JPG)、PNG以及 BMP 都是静态图片&#xff0c;而 GIF 则可以实现动态图片。 BMP(全称 Bitmap)是Window操作系统中的标准图像文件格式&am…