简介

java8也出来好久了,接口默认方法,lambda表达式,函数式接口,Date API等特性还是有必要去了解一下。比如在项目中经常用到集合,遍历集合可以试下lambda表达式,经常还要对集合进行过滤和排序,Stream就派上用场了。用习惯了,不得不说真的很好用。

Stream作为java8的新特性,基于lambda表达式,是对集合对象功能的增强,它专注于对集合对象进行各种高效、便利的聚合操作或者大批量的数据操作,提高了编程效率和代码可读性。

Stream的原理:将要处理的元素看做一种流,流在管道中传输,并且可以在管道的节点上处理,包括过滤筛选、去重、排序、聚合等。元素流在管道中经过中间操作的处理,最后由最终操作得到前面处理的结果。

集合有两种方式生成流:

  • stream() − 为集合创建串行流
  • parallelStream() - 为集合创建并行流

上图中是Stream类的类结构图,里面包含了大部分的中间和终止操作。

中间操作主要有以下方法(此类型方法返回的都是Stream):map (mapToInt, 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对对象集合的操作,新建一个Student类(学生类),覆写了equals()和hashCode()方法

public class Student {    private Long id;    private String name;    private int age;    private String address;    public Student() {}    public Student(Long id, String name, int age, String address) {        this.id = id;        this.name = name;        this.age = age;        this.address = address;    }    @Override    public String toString() {        return "Student{" +                "id=" + id +                ", name='" + name + ''' +                ", age=" + age +                ", address='" + address + ''' +                '}';    }    @Override    public boolean equals(Object o) {        if (this == o) return true;        if (o == null || getClass() != o.getClass()) return false;        Student student = (Student) o;        return age == student.age &&                Objects.equals(id, student.id) &&                Objects.equals(name, student.name) &&                Objects.equals(address, student.address);    }    @Override    public int hashCode() {        return Objects.hash(id, name, age, address);    }    public Long getId() {        return id;    }    public void setId(Long id) {        this.id = id;    }    public String getName() {        return name;    }    public void setName(String name) {        this.name = name;    }    public int getAge() {        return age;    }    public void setAge(int age) {        this.age = age;    }    public String getAddress() {        return address;    }    public void setAddress(String address) {        this.address = address;    }}

filter(筛选)

public static void main(String [] args) {        Student s1 = new Student(1L, "肖战", 15, "浙江");        Student s2 = new Student(2L, "王一博", 15, "湖北");        Student s3 = new Student(3L, "杨紫", 17, "北京");        Student s4 = new Student(4L, "李现", 17, "浙江");        List students = new ArrayList<>();        students.add(s1);        students.add(s2);        students.add(s3);        students.add(s4);        List streamStudents = testFilter(students);        streamStudents.forEach(System.out::println);    }    /**     * 集合的筛选     * @param students     * @return     */    private static List testFilter(List students) {        //筛选年龄大于15岁的学生//        return students.stream().filter(s -> s.getAge()>15).collect(Collectors.toList());        //筛选住在浙江省的学生        return students.stream().filter(s ->"浙江".equals(s.getAddress())).collect(Collectors.toList());    }

运行结果:

这里我们创建了四个学生,经过filter的筛选,筛选出地址是浙江的学生集合。

map(转换)

    public static void main(String [] args) {        Student s1 = new Student(1L, "肖战", 15, "浙江");        Student s2 = new Student(2L, "王一博", 15, "湖北");        Student s3 = new Student(3L, "杨紫", 17, "北京");        Student s4 = new Student(4L, "李现", 17, "浙江");        List students = new ArrayList<>();        students.add(s1);        students.add(s2);        students.add(s3);        students.add(s4);        testMap(students);    }    /**     * 集合转换     * @param students     * @return     */    private static void testMap(List students) {        //在地址前面加上部分信息,只获取地址输出        List addresses = students.stream().map(s ->"住址:"+s.getAddress()).collect(Collectors.toList());        addresses.forEach(a ->System.out.println(a));    }

运行结果

map就是将对应的元素按照给定的方法进行转换。

distinct(去重)

    public static void main(String [] args) {      testDistinct1();    }    /**     * 集合去重(基本类型)     */    private static void testDistinct1() {        //简单字符串的去重        List list = Arrays.asList("111","222","333","111","222");        list.stream().distinct().forEach(System.out::println);    }

运行结果:

public static void main(String [] args) {      testDistinct2();    }    /**     * 集合去重(引用对象)     */    private static void testDistinct2() {        //引用对象的去重,引用对象要实现hashCode和equal方法,否则去重无效        Student s1 = new Student(1L, "肖战", 15, "浙江");        Student s2 = new Student(2L, "王一博", 15, "湖北");        Student s3 = new Student(3L, "杨紫", 17, "北京");        Student s4 = new Student(4L, "李现", 17, "浙江");        Student s5 = new Student(1L, "肖战", 15, "浙江");        List students = new ArrayList<>();        students.add(s1);        students.add(s2);        students.add(s3);        students.add(s4);        students.add(s5);        students.stream().distinct().forEach(System.out::println);    }

运行结果:

可以看出,两个重复的“肖战”同学进行了去重,这不仅因为使用了distinct()方法,而且因为Student对象重写了equals和hashCode()方法,否则去重是无效的。

sorted(排序)

    public static void main(String [] args) {        testSort1();    }    /**     * 集合排序(默认排序)     */    private static void testSort1() {        List list = Arrays.asList("333","222","111");        list.stream().sorted().forEach(System.out::println);    }

运行结果:

    public static void main(String [] args) {        testSort2();    }    /**     * 集合排序(指定排序规则)     */    private static void testSort2() {        Student s1 = new Student(1L, "肖战", 15, "浙江");        Student s2 = new Student(2L, "王一博", 15, "湖北");        Student s3 = new Student(3L, "杨紫", 17, "北京");        Student s4 = new Student(4L, "李现", 17, "浙江");        List students = new ArrayList<>();        students.add(s1);        students.add(s2);        students.add(s3);        students.add(s4);        students.stream()                .sorted((stu1,stu2) ->Long.compare(stu2.getId(), stu1.getId()))                .sorted((stu1,stu2) -> Integer.compare(stu2.getAge(),stu1.getAge()))                .forEach(System.out::println);    }

运行结果:

上面指定排序规则,先按照学生的id进行降序排序,再按照年龄进行降序排序

limit(限制返回个数)

    public static void main(String [] args) {        testLimit();    }    /**     * 集合limit,返回前几个元素     */    private static void testLimit() {        List list = Arrays.asList("333","222","111");        list.stream().limit(2).forEach(System.out::println);    }

运行结果:

skip(删除元素)

    public static void main(String [] args) {        testSkip();    }    /**     * 集合skip,删除前n个元素     */    private static void testSkip() {        List list = Arrays.asList("333","222","111");        list.stream().skip(2).forEach(System.out::println);    }

运行结果:

reduce(聚合)

    public static void main(String [] args) {        testReduce();    }    /**     * 集合reduce,将集合中每个元素聚合成一条数据     */    private static void testReduce() {        List list = Arrays.asList("欢","迎","你");        String appendStr = list.stream().reduce("北京",(a,b) -> a+b);        System.out.println(appendStr);    }

运行结果:

min(求最小值)

    public static void main(String [] args) {        testMin();    }    /**     * 求集合中元素的最小值     */    private static void testMin() {        Student s1 = new Student(1L, "肖战", 14, "浙江");        Student s2 = new Student(2L, "王一博", 15, "湖北");        Student s3 = new Student(3L, "杨紫", 17, "北京");        Student s4 = new Student(4L, "李现", 17, "浙江");        List students = new ArrayList<>();        students.add(s1);        students.add(s2);        students.add(s3);        students.add(s4);        Student minS = students.stream().min((stu1,stu2) ->Integer.compare(stu1.getAge(),stu2.getAge())).get();        System.out.println(minS.toString());    }

运行结果:

上面是求所有学生中年龄最小的一个,max同理,求最大值。

anyMatch/allMatch/noneMatch(匹配)

    public static void main(String [] args) {        testMatch();    }    private static void testMatch() {        Student s1 = new Student(1L, "肖战", 15, "浙江");        Student s2 = new Student(2L, "王一博", 15, "湖北");        Student s3 = new Student(3L, "杨紫", 17, "北京");        Student s4 = new Student(4L, "李现", 17, "浙江");        List students = new ArrayList<>();        students.add(s1);        students.add(s2);        students.add(s3);        students.add(s4);        Boolean anyMatch = students.stream().anyMatch(s ->"湖北".equals(s.getAddress()));        if (anyMatch) {            System.out.println("有湖北人");        }        Boolean allMatch = students.stream().allMatch(s -> s.getAge()>=15);        if (allMatch) {            System.out.println("所有学生都满15周岁");        }        Boolean noneMatch = students.stream().noneMatch(s -> "杨洋".equals(s.getName()));        if (noneMatch) {            System.out.println("没有叫杨洋的同学");        }    }

运行结果:

anyMatch:Stream 中任意一个元素符合传入的 predicate,返回 true

allMatch:Stream 中全部元素符合传入的 predicate,返回 true

noneMatch:Stream 中没有一个元素符合传入的 predicate,返回 true

总结

上面介绍了Stream常用的一些方法,虽然对集合的遍历和操作可以用以前常规的方式,但是当业务逻辑复杂的时候,你会发现代码量很多,可读性很差,明明一行代码解决的事情,你却写了好几行。试试lambda表达式,试试Stream,你会有不一样的体验。

看完觉得不错可以关注一下小编,后续还会持续更新干货文章!!

java8 list 去重_Java8中的Stream,一行代码,让集合操作飞起来相关推荐

  1. Markdown中实现内容及代码块折叠操作

    在Markdown中实现内容及代码块折叠操作 代码: <details><summary>**展开查看源码**</summary>展开的内容及代码块 </de ...

  2. Java中lambda表达式去重_JAVA8中Lambda和Stream

    Java8于2014年3月份发布,其主要更新的特性有:函数式接口.Lambda 表达式.集合的流式操作.注解的更新.安全性的增强.IO\NIO 的改进.完善的全球化功能等,本文将介绍Lambda表达式 ...

  3. java8中-_java8中的Stream

    java8提出的函数式编程旨在帮助程序猿们写出更优雅的代码,上文函数式编程基础也介绍了java8新提出的一些函数式接口,通过它们代码貌似已经简洁了一波,但是,代码其实还可以更简洁下,接下来就要开始给大 ...

  4. java8遍历数组_java8新特性,使用流遍历集合

    在这篇"Java 8新特性教程"系列文章中,我们会深入解释,并通过代码来展示,如何通过流来遍历集合,如何从集合和数组来创建流,以及怎么聚合流的值. 在之前的文章"遍历.过 ...

  5. LeetCode中,python一行代码能干啥?

    导读 都说python语言简洁.集成高效,一行代码往往能实现很多复杂的操作,比如两变量交换.心形输出.打印乘法口诀等等.但这些总归还是不太实用.那么我们换做在LeetCode中,看看用python一行 ...

  6. java8实战怎么样_Java8中你可能不知道的一些地方之Stream实战

    说起流,我们会想起手机 ,电脑组装流水线,物流仓库商品包装流水线等等.如果把手机 ,电脑,包裹看做最终结果的话,那么加工商品前的各种零部件就可以看做数据源,而中间一系列的加工作业操作,就可以看做流的处 ...

  7. java8的表达式_java8中的Lambda表达式

    lambad表达式是一个匿名函数,既没有函数名的函数.在lambda表达式出现之前,java中更多使用的是匿名内部类所以有些刚刚接触lambda表达式的人会把lambda表达式误认为就是匿名内部类的简 ...

  8. java8时间间隔计算_Java8中Instant和LocalDate来计算时间或者日期间隔

    /** * java.time.Instant * java.time.Duration * Instant 默认使用UTC时区:2019-01-24T14:01:32.258Z * mongo中的时 ...

  9. 一行代码 实现集合去重

    前些天发现了一个巨牛的人工智能学习网站,通俗易懂,风趣幽默,忍不住分享一下给大家.点击跳转到教程. 1. 代码写法:(这是针对元素为基本数据类型的集合) 针对引用类型元素去重,可根据元素内多个元素去重 ...

最新文章

  1. 代码重构中的几个概念
  2. Recurrent Neural Network[Quasi RNN]
  3. 怎么求平均数_EXCEL怎么求企业连续几年业绩的平均增长率
  4. 数据库(三),底层算法
  5. python本地识别二维码_十行代码实现文字识别,你敢信?
  6. Linux-Shell脚本学习心得(第二天)
  7. Eclipse导入GitHub上面项目的两处报错处理
  8. 金字塔原理--公开演讲
  9. Data-Mediator入门系列4----常用类说明
  10. 高效科研神器——文献阅读篇
  11. AOSP 隐藏 su
  12. 微信公众号开放标签跳转小程序
  13. 零基础新人如何转行做新媒体运营?
  14. Minimum supported Gradle version is 6.1.1. Current version is 5.6.4
  15. C语言期末设计不在是千篇一律的学生管理系统,新题设值教师工作量管理系统(数组)。
  16. 什么软件可以代替sc防火墙_车玻璃水的成份是什么?普通肥皂水和清水可以代替吗?...
  17. 逆战网络连接断开连接服务器超时_《天天逆战》出现网络连接问题解决方案
  18. python unix时间戳_Python 获得13位unix时间戳的方法
  19. Automation Anywhere 认证考试
  20. 经验分享|原来这些图灵奖巨匠就藏在身边

热门文章

  1. Python 调试:pdb
  2. 软件工程的未来发展趋势[转载]
  3. 索引扫描时,对同一个叶子块访问多次的原因初探
  4. 今晚直播丨用高效的Oracle性能诊断工具,让运维工作轻松起来!
  5. 实战经验:关于Oracle Delete数据后空间重用问题的测试
  6. 强势回归丨2021数据库大咖讲坛(第1期):数据库高可用容灾方案的实践与探索
  7. 数据库周刊第十七期来啦!
  8. 源码编译安装PostgresSQL(11.5)
  9. 毕业季offer怎么拿?收下这份非典型求职面试指南
  10. 技术实践丨手把手教你使用MQTT方式对接华为IoT平台