0

0

Java8函数式编程之收集器怎么应用

王林

王林

发布时间:2023-05-14 19:46:09

|

1560人浏览过

|

来源于亿速云

转载

    收集器

    收集器是一种通用的、从流生成复杂值的结构。可以使用它从流中生成list、set、map等集合。 收集器都是在流的collect方法中调用,并且都在 collectors类中。

    java 的标准类库提供了很多有用的收集器,当然了,也可以自己自定义(这个对于使用者的要求很高)。

    下面提供一个代码,用于测试接下里要说的收集器:

    提供了一个简单的测试数据

    学号 姓名 性别 语文 数学 英语 物理 政治 总分09509002 节强 男 86 90 90 93 9009509003 杨青 女 90 90 82 91 9209509006 徐刚 男 78 92 83 90 8709509111 马力 男 77 88 99 90 8809509001 武向丽 女 90 78 83 94 9409509007 张文静 女 85 90 79 94 8809509005 徐小红 女 78 85 88 93 9209509009 李姝 女 92 80 75 90 8809509004 李文华 男 68 59 70 85 9009509008 夏婧 女 87 65 73 91 9509509010 王洪 男 66 48 89 70 57

    Student 实体类封装数据

    立即学习Java免费学习笔记(深入)”;

    package com.cdragon;
    
    public class Student implements Comparable<Student> {
        private String number;
        private String name;
        private String sex;
        private Integer chinese;
        private Integer math;
        private Integer english;
        private Integer physics;
        private Integer politics;
    	//省略getter和setter方法,这个使用IDE自动生成特别方便。
    	//省略toString方法,同上。
    	
    	@Override
        public int compareTo(Student s) {
            return s.getNumber().compareTo(this.getNumber());
        }
    }

    LoadData 类加载数据到内存

    package com.cdragon;
    
    import java.io.*;
    import java.util.ArrayList;
    import java.util.List;
    
    public class LoadData {
    
        public static List<Student> readFromFile(File file) {
            List<Student> students = new ArrayList<>();
            try (BufferedReader br = new BufferedReader(new InputStreamReader(new FileInputStream(file)))) {
                String record = null;
                String header = br.readLine();
                //对于数据的第一行头,暂时不做处理。
                while ((record = br.readLine() ) != null) {
                    Student s = resolveLineToStudent(record);
                    students.add(s);
                }
            } catch (FileNotFoundException e) {
                e.printStackTrace();
            } catch (IOException e) {
                e.printStackTrace();
            }
            return students;
        }
    
        private static Student resolveLineToStudent(String record) {
            String[] array = record.split("\s+");  // \s 和 \s+ 还是有区别的!
            Student s = new Student();
            s.setNumber(array[0]);
            s.setName(array[1]);
            s.setSex(array[2]);
            s.setChinese(Integer.parseInt(array[3]));
            s.setMath(Integer.parseInt(array[4]));
            s.setEnglish(Integer.parseInt(array[5]));
            s.setPhysics(Integer.parseInt(array[6]));
            s.setPolitics(Integer.parseInt(array[7]));
            return s;
        }
    
    }

    收集器应用

    将流转换成其他集合

    使用收集器是可以生成其他集合的,例如生成List、Set 和 Map等,下面来分别举例:

    		//生成 List
            List<Student> studentList = students.stream().collect(Collectors.toList());
            //生成 Set
            Set<Student> studentSet = students.stream().collect(Collectors.toSet());
            //生成指定集合
            TreeSet<Student> studentTreeSet = students.stream().collect(Collectors.toCollection(TreeSet::new));
            //生成 Map 
            Map<String, Student> studentMap = students.stream().collect(Collectors.toMap(Student::getNumber, s->s));
    
            studentMap.forEach((no, s)->{
                System.out.println(no + "->" + s);
            });

    说明:通常的 toList() 和 toSet() 方法是不指定生成集合的具体类型,这是由系统来选择最合适的类型,但是有时候我们必须返回特定的类型集合,这就用到了 toCollection() 方法,这个方法可以指定需要生成的集合的类型,这是使用方法引用进行简化代码:TreeSet::new

    测试结果:

    Java8函数式编程之收集器怎么应用

    注意:生成 Map 的方式较为复杂,因为需要同时指定键和值。

    转换成值

    使用收集器生成一个值。

    最大值和最小值

    Collectors 类中的 maxBy 和 minBy 允许用户按照某种特定顺序生成一个值。它们的作用就如同它们的名字一样,分别是寻找最大值和最小值。

    我写成一个方法,这样调用比较方便。

    /**
     * 获取单科最高分。
     * */
    public static Optional<Student> minOrMaxSubject(List<Student> students, Comparator<? super Student> comparator) {
        return students.stream().collect(Collectors.maxBy(comparator));
    }

    说明:使用 maxBy 或者 minBy 必须传入一个 Comparator 对象作为参数,即参数为一个比较器。

    测试代码

    //这个文件的路径应该使用自己指定的
    List<Student> students = LoadData.readFromFile(new File("src/grade.txt"));
    //获取数学最高分学生
    Optional<Student> s1 = TestStream.minOrMaxSubject(students, Comparator.comparing(Student::getMath));
    //获取英语最高分
    Optional<Student> s2 = TestStream.minOrMaxSubject(students, Comparator.comparing(Student::getEnglish));
    System.out.println(s1.get());
    System.out.println(s2.get());

    测试结果

    VIVA
    VIVA

    一个免费的AI创意视觉设计平台

    下载

    Java8函数式编程之收集器怎么应用

    说明:如果想要测试最低分,只要把上面的 maxBy 改成 minBy 就行了,或者直接更进一步,修改参数为collect里面传入的函数,不过那样就会显得格外复杂,而且不止可以查最高分和最低分了。

    平均值

    上面看过了最大值和最小值,现在来看看平均值。 下面这个方法是用来求单科平均分的。

    /**
     * 获取单科平均分
     * */
     public static double averageScore(List<Student> students, ToIntFunction<? super Student> mapper) {
         return students.stream().collect(Collectors.averagingInt(mapper));
     }

    测试代码

    List<Student> students = LoadData.readFromFile(new File("src/grade.txt"));
    double math = TestStream.averageScore(students, Student::getMath);
    System.out.println("数学的单科平均分:" + math);

    测试结果

    Java8函数式编程之收集器怎么应用

    数据分块

    数据分块是指收集器将流分为两个集合,注意分块是只能分成两块。这里标准类库提供了一个收集器 partitioningBy,它接受一个流,并将其分为两个部分。返回的结果为一个 Map,键只有两种:true 或者 false,值是满足对应条件的集合。

    例如我想知道某们成绩 90分以上和一下的学生分别是哪些。

    /**
     * 以特定分数划分不同学生,例如90分以上(含90分)和90分一下。
     * 结果是一个Map集合,只有两个元素,true  false 个对应一个集合。
     * */
     public static Map<Boolean, List<Student>> splitScore(List<Student> students, Predicate<? super Student> predicate) {
         return students.stream().collect(Collectors.partitioningBy(predicate));
     }

    说明:partitioningBy的参数为一个 Predicate 对象,这个和过滤器的很相似,功能上可以对比学习。

    测试代码

    //数学成绩以90分来划分学生
    Map<Boolean, List<Student>> booleanListMap = TestStream.splitScore(students, stu->stu.getMath()>=90);
    booleanListMap.forEach((bool, list)->{
        System.out.println("数学成绩大于90分:" + bool);
        list.forEach(System.out::println);
        System.out.println("========================");
    });

    测试结果

    Java8函数式编程之收集器怎么应用

    数据分组

    数据分组是一种更为自然的分割数据操作,与将数据分成true和false两部分不同,可以使用任意值对数据分组。比如使用性别对学生进行分组。这很像SQL中的 groupBy 操作。

    /**
    * 数据分组
    * 这里以性别来分组
    * */
    public static Map<String, List<Student>> groupBy(List<Student> students) {
        return students.stream().collect(Collectors.groupingBy(Student::getSex));
    }

    测试代码

    Map<String, List<Student>> stringListMap = TestStream.groupBy(students);
    stringListMap.forEach((sex, list)->{
        System.out.println("性别:" + sex);
        list.forEach(System.out::println);
        System.out.println("============");
    });

    测试结果

    Java8函数式编程之收集器怎么应用

    字符串

    收集流中的数据最后生成一个字符串,这是一个很平常的操作。 例如一个所有学生的姓名列表,使用传统的迭代列表操作代码如下:

     /**
      * 获取所有学生姓名的字符串
      * 传统的迭代操作
      * 格式如下:[张三,李四]
      * */
     public static String nameStr1(List<Student> students) {
         StringBuilder builder = new StringBuilder("[");
         for (Student stu : students) {
             if (builder.length() > 1){
                 builder.append(",");
             }
             String name = stu.getName();
             builder.append(name);
         }
         builder.append("]");
         return builder.toString();
     }

    然后是使用收集器进行操作,代码如下: 这里我添加一些细节处理,学生的排名按照学生的总成绩从高到底排列,这是很符合习惯的。

    /**
     * 获取所有学生姓名的字符串
     * 函数式方法
     * 格式如下:[张三,李四]
     *
     * 注意,他只能连接字符串,所有这里使用 map 操作,将 Student 转成 String(学生姓名)
     * */
    public static String nameStr2(List<Student> students) {
        return students.stream()
                .sorted(Comparator.comparing(s -> {
                    return s.getChinese() + s.getMath() + s.getEnglish()
                            + s.getPhysics() + s.getPolitics();
                }, Comparator.reverseOrder()))  //(sum1, sum2)-> sum2.compareTo(sum1)
                .map(Student::getName)
                .collect(Collectors.joining(",","[","]"));
    }

    说明:这里的 sorted 需要传入一个 Comparator 对象,但是可以使用静态方法 Compring 进行简化,但是它只是指定需要排序的标准,并没有说是从小到大还是从大到小,后来才发现,这个是默认的:大小到大排序。但是我需要的是使用从大到小,然后发现原来 compring 还有重载方法,具有两个参数,另一个参数是可以指定大小顺序的,所以第二个参数我传入了一个 Lambda 表达式:

    (sum1, sum2)-> sum2.compareTo(sum1)

    但是如果这样使用的话,还不如直接使用 Lambda 表达式创建 Comparator 对象方便呢,后来发现这个 IDE 比较智能,它指出这句代码,可以被替换为:

    Comparator.reverseOrder();// 看意思就知道是 反序的意思。

    这样看来使用 Comparator 静态的 comparing 方法还是比直接创建 Comparator 对象简单一些。

    注意:如果不需要排序的话,就只有一个map方法和join方法了。这个map方法的作用是映射(我一开始把它和map集合总是搞混了),将Student对象映射为name字符串,然后使用 join 方法进行连接。

    组合收集器

    收集器还可以组合起来使用,这个和 SQL 感觉更像了,几乎具有函数式编程的语言,都有SQL那种处理数据的方式,例如最大值、最小值和分组等操作。 考虑对于学生按照性别分组,然后再分别统计男女生的人数。(这个在 SQL 里面也是一个基本的练习。)

    /**
     * 组合收集器
     * 这里以性别来分组,再分别计数
     * */
    public static Map<String, Long> combination(List<Student> students) {
        return students.stream().collect(Collectors.groupingBy(Student::getSex, Collectors.counting()));
    }

    测试代码

    Map<String, Long> stringLongMap = TestStream.combination(students);
    stringLongMap.forEach((sex, count)->{
        System.out.println("性别:" + sex + ", 人数:" + count);
    });

    测试结果

    Java8函数式编程之收集器怎么应用

    使用流的其他操作

    对于流的使用,应该达到一个较为熟练的地步,但是由于没有什么机会实践,还是比较陌生。下面介绍几个我写的方法,来看看流的操作:

    //通过过滤器选择特定的学生,过滤器用于过滤,然后选择第一个学生。
    //这里应该加一个排序操作比较好。
    public static Optional<Student> selectStudent(List<Student> students, Predicate<?  super Student> pre) {
            return students.stream().filter(pre).findFirst();
        }
    
    public static List<Student> orderBy(List<Student> students, Comparator<? super Student> comparator) {
        if (comparator != null){
            return students.stream().sorted(comparator).collect(Collectors.toList());
        } else {
            return students.stream().sorted().collect(Collectors.toList());
        }
    }
    
    //获取一列数据。不是一行学生记录,是一列。
    public static List<?> getAColumn(List<Student> students, Function<? super Student, ?> mapper) {
        return students.stream().map(mapper).collect(Collectors.toList());
    }
    
    /**
     * 获取所有学生的总分和学号
     * */
    public static Map<String, Integer> getSum(List<Student> students) {
        return students.stream().collect(Collectors.toMap(Student::getNumber, stu->{
            return stu.getChinese() + stu.getEnglish() + stu.getMath() + stu.getPhysics() + stu.getPolitics();
        }));
    }
    
    
    /**
     * peek 和 forEach 的区别
     * peek 是一个中间操作,forEach 是一个终结操作。
     *
     * 假如实现一个功能:每个学生的某门科目分数进行修改。
     *
     * peek 操作后得到的仍然是一个 stream,此时可以进一步操作,
     * 但是 forEach 是终结操作,操作结束,流就结束了,如果需要进一步处理,
     * 必须再次进行得到流的操作。
     * */
    
    public static List<Student> addScore1(List<Student> students, Consumer<? super Student> action) {
        return students.stream().peek(action).collect(Collectors.toList());
    }
    
    public static void addScore2(List<Student> students, Consumer<? super Student> action) {
        students.stream().forEach(action);
    }
    
    //指定返回类型为 LinkedList,这时一个测试,并不是说需要这样写。
    //多数情况下,我们还是应该使用 ArrayList
    public static List<Student> addScore3(List<Student> students, Consumer<? super Student> action) {
        return students.stream().peek(action).collect(Collectors.toCollection(LinkedList::new));
    }

    对于其中的几个进行测试(不是全部方法,如果感兴趣,可以自己尝试。):

    //对于学生进行排序,参数为一个比较器,参数为空的话,使用默认的 sorted 排序。
    //测试代码 按照学号排序(默认从小到大)
    TestStream.orderBy(students,Comparator.comparing(Student::getNumber)).forEach(System.out::println);
    
    //按照学号排序(从大到小)
    TestStream.orderBy(students,Comparator.comparing(Student::getNumber, Comparator.reverseOrder())).forEach(System.out::println);
    
    //使用默认的排序
    TestStream.orderBy(students).forEach(System.out::println);
    
    //获取一列学生的记录,例如这里是英语成绩,这里返回值我使用通配符应该没有错吧
    //因为返回数据可能为 String 也可能是 Integer
    TestStream.getAColumn(students,Student::getEnglish).forEach(System.out::println);
    
    //测试学生的总分
    TestStream.getSum(students).forEach((no, stu)->{
        System.out.println(no + " -> " + stu);
    });

    热门AI工具

    更多
    DeepSeek
    DeepSeek

    幻方量化公司旗下的开源大模型平台

    豆包大模型
    豆包大模型

    字节跳动自主研发的一系列大型语言模型

    通义千问
    通义千问

    阿里巴巴推出的全能AI助手

    腾讯元宝
    腾讯元宝

    腾讯混元平台推出的AI助手

    文心一言
    文心一言

    文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

    讯飞写作
    讯飞写作

    基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

    即梦AI
    即梦AI

    一站式AI创作平台,免费AI图片和视频生成。

    ChatGPT
    ChatGPT

    最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

    相关专题

    更多
    数据分析工具有哪些
    数据分析工具有哪些

    数据分析工具有Excel、SQL、Python、R、Tableau、Power BI、SAS、SPSS和MATLAB等。详细介绍:1、Excel,具有强大的计算和数据处理功能;2、SQL,可以进行数据查询、过滤、排序、聚合等操作;3、Python,拥有丰富的数据分析库;4、R,拥有丰富的统计分析库和图形库;5、Tableau,提供了直观易用的用户界面等等。

    1133

    2023.10.12

    SQL中distinct的用法
    SQL中distinct的用法

    SQL中distinct的语法是“SELECT DISTINCT column1, column2,...,FROM table_name;”。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

    340

    2023.10.27

    SQL中months_between使用方法
    SQL中months_between使用方法

    在SQL中,MONTHS_BETWEEN 是一个常见的函数,用于计算两个日期之间的月份差。想了解更多SQL的相关内容,可以阅读本专题下面的文章。

    381

    2024.02.23

    SQL出现5120错误解决方法
    SQL出现5120错误解决方法

    SQL Server错误5120是由于没有足够的权限来访问或操作指定的数据库或文件引起的。想了解更多sql错误的相关内容,可以阅读本专题下面的文章。

    2152

    2024.03.06

    sql procedure语法错误解决方法
    sql procedure语法错误解决方法

    sql procedure语法错误解决办法:1、仔细检查错误消息;2、检查语法规则;3、检查括号和引号;4、检查变量和参数;5、检查关键字和函数;6、逐步调试;7、参考文档和示例。想了解更多语法错误的相关内容,可以阅读本专题下面的文章。

    380

    2024.03.06

    oracle数据库运行sql方法
    oracle数据库运行sql方法

    运行sql步骤包括:打开sql plus工具并连接到数据库。在提示符下输入sql语句。按enter键运行该语句。查看结果,错误消息或退出sql plus。想了解更多oracle数据库的相关内容,可以阅读本专题下面的文章。

    1663

    2024.04.07

    sql中where的含义
    sql中where的含义

    sql中where子句用于从表中过滤数据,它基于指定条件选择特定的行。想了解更多where的相关内容,可以阅读本专题下面的文章。

    585

    2024.04.29

    sql中删除表的语句是什么
    sql中删除表的语句是什么

    sql中用于删除表的语句是drop table。语法为drop table table_name;该语句将永久删除指定表的表和数据。想了解更多sql的相关内容,可以阅读本专题下面的文章。

    440

    2024.04.29

    C# ASP.NET Core微服务架构与API网关实践
    C# ASP.NET Core微服务架构与API网关实践

    本专题围绕 C# 在现代后端架构中的微服务实践展开,系统讲解基于 ASP.NET Core 构建可扩展服务体系的核心方法。内容涵盖服务拆分策略、RESTful API 设计、服务间通信、API 网关统一入口管理以及服务治理机制。通过真实项目案例,帮助开发者掌握构建高可用微服务系统的关键技术,提高系统的可扩展性与维护效率。

    3

    2026.03.11

    热门下载

    更多
    网站特效
    /
    网站源码
    /
    网站素材
    /
    前端模板

    精品课程

    更多
    相关推荐
    /
    热门推荐
    /
    最新课程
    Kotlin 教程
    Kotlin 教程

    共23课时 | 4.3万人学习

    C# 教程
    C# 教程

    共94课时 | 11.1万人学习

    Java 教程
    Java 教程

    共578课时 | 80.5万人学习

    关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
    php中文网:公益在线php培训,帮助PHP学习者快速成长!
    关注服务号 技术交流群
    PHP中文网订阅号
    每天精选资源文章推送

    Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号