我发现,自从我学了Stream流式操作之后,工作中使用到的频率还是挺高的,因为stream配合着lambda表达式或者双冒号(::)使用真的是优雅到了极致!今天就简单分(搬)享(运)一下我对stream流式操作的一点理解

一、什么是流式操作?或者准确的说什么是“流”

所谓艺术来源于生活,所以不妨从一个生活中小例子展开说明一下。

通过组词,流可以组成流动,流水,流经,流进流出.... 这些描述的都是一种状态,是一种“运动”中的状态,更加通俗的讲,流水线。以往的流水线会有很多工人在,每一个工人负责一个环节,环环相联下来之后最终会形成成品。好了,现在分解一下

  1. 首先需要有一条流水线(好给你流水线)

  2. 流水线有了,那么总得需要有人在作业吧,那就放几个卡通人上去充当一下,他们将完成对原材料的筛选、清洗、组装、贴标签等操作,注意这是有顺序,这也很好理解,下一道工序必须是在上一道工序的基础上进行的,总不能跳过组装就直接装箱吧,虽然是可以,但是你发个空包给别人,真的好吗?

    所以流水线+作业人员有了

  3. 经过一番操作之后,产出产品,之后就需要打包出厂,走进寻常百姓家了。最终完整的流程就是酱紫的

所以,对于“流式操作”可以想象成把一项需要完成的操作“打扁”成原材料,原材料会经过经过很多道工序,而经过多道工序时的状态就是“流”,最终会对流进行一个打包收集,形成业务产品。

二、对于stream,你不得不了解的几个概念

第一部分通过一个简单的例子,阐述了一下先决概念,接下来就就对stream本身进行了解。

中间操作符与终止操作符

一句话讲明,中间操作符可以有多个,每个中间操作符都是在前一个中间操作符的基础上进行操作的,这就像极了刚刚讲解的多道工序,二终止操作符有且仅有一个,也即处理完所有的中间操作之后,对最终产品进行收集或者消费的,流在这里之后就会得到一个实际的“产品”。

常见的中间操作符

操作符 对应的概念
flatmap(flatmapToInt,flatmapToLong,flatmapToDouble) 拍平操作 比如把 int[]{2,3,4} 拍平 变成 2,3,4 也就是从原来的一个数据变成了3个数据,这里默认提供了拍平成int,long,double的操作符
limit 限流操作 比如数据流中有10个 我只要出前3个就可以使用
distint 去重操作 对重复元素去重,底层使用了equals方法
filter 过滤操作 把不想要的数据过滤
peek 挑出操作 如果想对数据进行某些操作,如:读取、编辑修改等
skip 跳过操作 跳过某些元素
sorted(unordered) 排序操作 对元素排序,前提是实现Comparable接口,当然也可以自定义比较器

常见的终止操作符

操作符 对应的概念
collect 收集 使用系统提供的收集器可以将最终的数据流收集到List,Set,Map等容器中
count 统计操作 统计最终的数据个数
findFirst、findAny 查找操作 查找第一个、查找任何一个 返回的类型为Optional
noneMatch、allMatch、anyMatch 匹配操作 数据流中是否存在符合条件的元素 返回值为bool 值
min、max 最值操作,需要自定义比较器 返回数据流中最大最小的值
reduce 规约操作 将整个数据流的值规约为一个值,count、min、max底层就是使用reduce
forEach、forEachOrdered 遍历操作 这里就是对最终的数据进行消费了
toArray 数组操作 将数据流的元素转换成数组

️所有的这些操作都是建立在流的基础上,所以首先需要把对象打成流,而stream主要用于对象类型的集合,当然基础类型的stream其实也是有提供的,这是打成流的方式不一致,待会会展示到底哪里不一致了。

三、stream的简单小操作展示

@Data
@AllArgsConstructor
@NoArgsConstructor
@ToString
class Person{
private Integer id;
private String name;
private Integer age;
private String sex;
}
/**
* @author Amg
* @Description 首先展现一下对于对象的集合跟基本类型是怎么打成流的
* @param
* @date 2020/9/13
* @return void
*/
public void tt(){

   //基于对象的集合
   List<Person> list = new ArrayList<>();
   Stream<Person> collection = list.stream();

   //基于基本类型的
   int[] array = new int[]{1,2,3,4,5,6,7,8,9};
   IntStream primitive = Arrays.stream(array);

   //挖一下源码就会发现,其实Stream<T>跟IntStream都是继承自BaseStream的,有Int自然也会有Double、Long等等
}

/**
* @author Amg
* @Description 第二个讲一下遍历过滤输出一个给定的范围的偶数
* @param
* @date 2020/9/13
* @return void
*/
public void testStream01() {
   IntStream.range(0, 20).filter(i -> (i % 2 == 0)).forEach(result -> System.out.print(result + " "));
}

//output 0 2 4 6 8 10 12 14 16 18 这里得注意了,对于lambda表达式跟方法引用的使用大家是要会的
/**
* @author Amg
* @Description 模拟场景就是一个带重复数据的数组集合,把它转换成一个去重后的List集合并且打印List中所有元素
* @param
* @date 2020/9/13
* @return void
*/
public void testStream02(){

   Object[] array = new Object[]{1,1,1,1,1,1,2,2,2,
                                 2,2,2,3,3,3,3,3,
                                 'a','a','a','a',
                                 "e","e","e","e",
                                 3.579f,3.579f,594d,594d,
                                 97586L,97586L
                                };
   Arrays.stream(array).distinct().collect(Collectors.toList()).forEach(res-> System.out.print(res + " "));

   //output 1 2 3 a e 3.579 594.0 97586 distinct就是去重的操作
}
/**
* @author Amg
* @Description 这次模拟的场景是对对象进行操作,把list转换成map,map的键是对象的id主键,值是整个对象
* @param
* @date 2020/9/13
* @return void
*/
public void testStream03(){

   //prepareData()准备一些数据
   List<Person> list = prepareData();
   System.out.println(list);
   //Function.identify() 等价于 t -> t 返回自身
   //toMap的第三个参数可以解决冲突问题,当键值都是一致的情况,如果不添加第三个参数就会抛异常,下面的写法代表,如果有冲突则保留其中一个
   Map<Integer, Person> result = list.stream().collect(Collectors.toMap(Person -> Person.getId() ,Function.identity(),(entity1,entity2)-> entity1));
   result.forEach((key,value)->{
       System.out.println(key + ":[" + value + "]");
  });
}

//output
/**  
[Person(id=1, name=z3, age=3, sex=男), Person(id=2, name=l4, age=4, sex=男), Person(id=3, name=w5, age=2, sex=男), Person(id=4, name=z6, age=13, sex=女), Person(id=5, name=t7, age=35, sex=女), Person(id=5, name=t7, age=35, sex=女)]
1:[Person(id=1, name=z3, age=3, sex=男)]
2:[Person(id=2, name=l4, age=4, sex=男)]
3:[Person(id=3, name=w5, age=2, sex=男)]
4:[Person(id=4, name=z6, age=13, sex=女)]
5:[Person(id=5, name=t7, age=35, sex=女)]

*/
/**
* @author Amg
* @Description 再考虑一个业务,从数据库中获取到对应表中指定的数据集,然后我们需要的是该表的主键id值,
* 然后通过这个主键id值去查,另外一张表的数据,得到的数据才是我们真实需要的
* @param
* @date 2020/9/13
* @return void
*/
public void testStream04(){

   //模拟环境,数据直接从方法里面模拟
   List<Person> list = getAllPersonById();

   //原操作是如此的
   List<Integer> personId = new ArrayList<>();
   for (Person person : list) {
       personId.add(person.getId());
  }

   //list里面存放的是所有的Person对象,我们将通过Stream流操作抽取出我们需要id,一行代码可以完成
   List<Integer> result = list.stream().map(Person::getId).collect(Collectors.toList());
   System.out.println(result);
   
   //output [1, 2, 3, 4, 5, 5]
}

好了,这就是今天分享的小内容,感谢你的观看,祝你生活愉快!

Java8——Stream流式操作的一点小总结的更多相关文章

  1. Java8中的Stream流式操作 - 入门篇

    作者:汤圆 个人博客:javalover.cc 前言 之前总是朋友朋友的叫,感觉有套近乎的嫌疑,所以后面还是给大家改个称呼吧 因为大家是来看东西的,所以暂且叫做官人吧(灵感来自于民间流传的四大名著之一 ...

  2. Java的Stream流式操作

    前言 最近在实习,在公司看到前辈的一些代码,发现有很多值得我学习的地方,其中有一部分就是对集合使用Stream流式操作,觉得很优美且方便.所以学习一下Stream流,在这里记录一下. Stream是什 ...

  3. 《JAVA8开发指南》使用流式操作

    为什么需要流式操作 集合API是Java API中最重要的部分.基本上每一个java程序都离不开集合.尽管很重要,但是现有的集合处理在很多方面都无法满足需要. 一个原因是,许多其他的语言或者类库以声明 ...

  4. Java1.8新特性 - Stream流式算法

    一. 流式处理简介   在我接触到java8流式数据处理的时候,我的第一感觉是流式处理让集合操作变得简洁了许多,通常我们需要多行代码才能完成的操作,借助于流式处理可以在一行中实现.比如我们希望对一个包 ...

  5. 【转】Java8 Stream 流详解

      当我第一次阅读 Java8 中的 Stream API 时,说实话,我非常困惑,因为它的名字听起来与 Java I0 框架中的 InputStream 和 OutputStream 非常类似.但是 ...

  6. java1.8新特性之stream流式算法

    在Java1.8之前还没有stream流式算法的时候,我们要是在一个放有多个User对象的list集合中,将每个User对象的主键ID取出,组合成一个新的集合,首先想到的肯定是遍历,如下: List& ...

  7. 第46天学习打卡(四大函数式接口 Stream流式计算 ForkJoin 异步回调 JMM Volatile)

    小结与扩展 池的最大的大小如何去设置! 了解:IO密集型,CPU密集型:(调优)  //1.CPU密集型 几核就是几个线程 可以保持效率最高 //2.IO密集型判断你的程序中十分耗IO的线程,只要大于 ...

  8. Stream流式编程

    Stream流式编程   Stream流 说到Stream便容易想到I/O Stream,而实际上,谁规定“流”就一定是“IO流”呢?在Java 8中,得益于Lambda所带来的函数式编程,引入了一个 ...

  9. Stream流式计算

    Stream流式计算 集合/数据库用来进行数据的存储 而计算则交给流 /** * 现有5个用户,用一行代码 ,一分钟按以下条件筛选出指定用户 *1.ID必须是偶数 *2.年龄必须大于22 *3.用户名 ...

随机推荐

  1. ACM study day3

    今天练了二分和快速幂,题目挺难的,挑几个我做上的说一下吧. 先给出几个二分和快速幂的模板函数: 二分 void BS(int m) { int x=,y=a[m-]-a[]; while(y-x> ...

  2. 区块链入门到实战(27)之以太坊(Ethereum) – 智能合约开发

    智能合约的优点 与传统合同相比,智能合约有一些显著优点: 不需要中间人 费用低 代码就是规则 区块链网络中有多个备份,不用担心丢失 避免人工错误 无需信任,就可履行协议 匿名履行协议 以太坊(Ethe ...

  3. C语言基础练习——打印菱形

    C语言基础练习--打印菱形 JERRY_Z. ~ 2020 / 8 / 26 转载请注明出处! 代码: /* * @Author: JERRY_Z. * @Date: 2020-08-26 17:17 ...

  4. 从零开始的SpringBoot项目 ( 三 ) 项目打包( war包篇 )

    pom.xml 修改打包类型 jar 改为 war 添加 tomcat 依赖 找到最右边的 Maven Projects,点击进去,选择需要打包的项目,并点击 install,就开始打包了,打包前先点 ...

  5. IDEA2019.3激活

    1      下载: 1.1    下载安装文件 https://blog.csdn.net/Beuty_Chen/article/details/103804886 1.2    下载激活补丁 链接 ...

  6. Mac 系统安装robot framework

    1.安装Python3 版本 2.安装robotframework:pip3 install robotframework 3.安装Pypubsub:pip3 install Pypubsub 4.安 ...

  7. Qt 怎样生成带图标的exe

    一.问题描述 当我们在 Windows 下用 VS 生成 exe 程序时,如果窗口程序指定了图标,那么生成的 exe 程序便是指定的图标模样. 但是,当使用 Qt Creator 编译程序却不同.即使 ...

  8. HttPServletRequest 对象 基本应用

    HttPServletRequest 对象 基本应用 防盗链 获取Referer这个消息头,判断Referer是不是从指定页面来的.如果不是从指定页面的,跳转回指定页面. 这可以用于保证页面广告的观看 ...

  9. Selenium多浏览器处理

    当我们在执行自动化测试过程中,往往会针对不同的浏览器做兼容性测试,那么我们在代码中,可以针对执行命令传过来的参数,选择对应的浏览器来执行测试用例 代码如下: 在终端中执行命令如上图红框中所示: bro ...

  10. [BUUOJ记录] [ACTF2020 新生赛]Upload

    简单的上传题,考察绕过前端Js验证,phtml拓展名的应用 打开题目点亮小灯泡后可以看到一个上传点 传一个php测试一下: 发现有文件拓展名检查,F12发现是Js前端验证: 审查元素直接删掉,继续上传 ...