• collect 详细解析 Java 8 Stream API 中的 collect 方法


    collect 详解

     Java 8 引入的 Stream API 提供了一种强大的方式来处理集合数据
     其中的 collect 方法是一个关键操作,用于将流中的元素收集到不同类型的结果容器中。
     本文将详细介绍 JavaStream 的 collect 方法
     包括其基本用法、常见的收集器以及一些实际应用场景。
    

    1,collect 方法的基本用法

    collect 方法是 Stream API 中的一个终端操作,它接受一个 Collector 参数,用于将流中的元素累积成一个结果。Collector 接口定义了如何收集元素的方法,通常使用 Collectors 工具类来提供各种预定义的收集器。

    基本语法:

    <R, A> R collect(Collector<? super T, A, R> collector)
    
    其中:
    T 是流中元素的类型。
    A 是累加器的类型,用于在收集过程中累积部分结果。
    R 是收集操作最终要返回的类型。
    

    示例1:将元素收集为列表

    假设我们有一个字符串列表,我们希望将其中长度大于等于5的字符串收集到一个新的列表中。

    List<String> words = Arrays.asList("apple", "banana", "grape", "orange", "kiwi");
    
    List<String> longWords = words.stream()
                                 .filter(word -> word.length() >= 5)
                                 .collect(Collectors.toList());
    
    System.out.println(longWords); // 输出:[banana, orange]
    

    在这个示例中,collect(Collectors.toList()) 使用了 Collectors 工具类提供的静态方法 toList(),将流中符合条件的元素收集到一个新的 List 中。

    当使用Java中的Stream API时,collect 方法是一个非常有用的工具,用于将流中的元素收集到不同类型的结果容器中。以下是一些直观易懂的例子,展示了 collect 方法的不同用法和常见场景。

    假设我们有一个整数列表,我们想要将其中的偶数收集到一个新的列表中:

    import java.util.Arrays;
    import java.util.List;
    import java.util.stream.Collectors;
    
    public class CollectExamples {
    
        public static void main(String[] args) {
            List<Integer> numbers = Arrays.asList(1, 2, 3, 4, 5, 6, 7, 8, 9, 10);
    
            List<Integer> evenNumbers = numbers.stream()
                                               .filter(n -> n % 2 == 0)
                                               .collect(Collectors.toList());
    
            System.out.println("Even numbers: " + evenNumbers); // 输出:Even numbers: [2, 4, 6, 8, 10]
        }
    }
    

    在这个例子中,.collect(Collectors.toList()) 将流中的偶数收集到一个新的列表中。

    示例2:将元素收集为 Set

    假设我们有一个字符串列表,我们想要将其中的元素收集到一个 HashSet 中:

    import java.util.Arrays;
    import java.util.HashSet;
    import java.util.List;
    import java.util.Set;
    import java.util.stream.Collectors;
    
    public class CollectExamples {
    
        public static void main(String[] args) {
            List<String> words = Arrays.asList("apple", "banana", "orange", "apple", "grape");
    
            Set<String> uniqueWords = words.stream()
                                          .collect(Collectors.toSet());
    
            System.out.println("Unique words: " + uniqueWords); // 输出:Unique words: [orange, grape, banana, apple]
        }
    }
    

    在这个例子中,.collect(Collectors.toSet()) 将列表中的字符串元素收集到一个 HashSet 中,去除了重复的元素。

    示例3:将元素收集为 Map

    假设我们有一个对象列表,每个对象有一个唯一的 ID 和一个名称,我们希望将这些对象收集到一个 Map 中,以对象的 ID 作为键,对象本身作为值:

    import java.util.Arrays;
    import java.util.List;
    import java.util.Map;
    import java.util.stream.Collectors;
    
    class Item {
        private int id;
        private String name;
    
        public Item(int id, String name) {
            this.id = id;
            this.name = name;
        }
    
        public int getId() {
            return id;
        }
    
        public String getName() {
            return name;
        }
    }
    
    public class CollectExamples {
    
        public static void main(String[] args) {
            List<Item> items = Arrays.asList(
                new Item(1, "Apple"),
                new Item(2, "Banana"),
                new Item(3, "Orange")
            );
    
            Map<Integer, Item> itemMap = items.stream()
                                             .collect(Collectors.toMap(Item::getId, item -> item));
    
            System.out.println("Item map: " + itemMap); 
            // 输出:Item map: {1=Item{id=1, name='Apple'}, 2=Item{id=2, name='Banana'}, 3=Item{id=3, name='Orange'}}
        }
    }
    

    在这个例子中,.collect(Collectors.toMap(Item::getId, item -> item)) 将对象列表转换为一个 Map,其中键为对象的 ID,值为对象本身。

    示例4:自定义收集器

    有时候,我们需要根据特定的需求创建自定义的收集器。例如,假设我们希望将列表中的字符串连接成一个以逗号分隔的字符串:

    import java.util.Arrays;
    import java.util.List;
    import java.util.stream.Collectors;
    
    public class CollectExamples {
    
        public static void main(String[] args) {
            List<String> letters = Arrays.asList("a", "b", "c", "d");
    
            String result = letters.stream()
                                   .collect(Collectors.joining(", "));
    
            System.out.println("Concatenated string: " + result);
             // 输出:Concatenated string: a, b, c, d
        }
    }
    

    在这个例子中,.collect(Collectors.joining(", ")) 使用了 joining 方法来将流中的字符串连接成一个以逗号分隔的单个字符串。

    这些例子展示了 collect 方法在日常编程中的实际应用,通过合适的收集器,可以轻松地将流中的元素转换为我们想要的数据结构或格式。

    2,常见的收集器

    Java 提供了丰富的预定义收集器,通过 Collectors 工具类可以方便地使用这些收集器。以下是一些常见的收集器示例:

    将元素收集为 Set

    Set<String> wordSet = words.stream()
                               .collect(Collectors.toSet());
    
    System.out.println(wordSet);
     // 输出:[apple, banana, grape, orange, kiwi]
    

    将元素收集为 Map
    假设我们有一个对象列表,我们想要将列表中的对象按照某个属性收集到一个 Map 中,以属性值作为键,对象列表作为值。

    class Person {
        private String name;
        private int age;
    
        // 省略构造方法和其他方法
    }
    
    List<Person> people = Arrays.asList(
        new Person("Alice", 30),
        new Person("Bob", 25),
        new Person("Charlie", 35)
    );
    
    Map<String, Person> nameToPersonMap = people.stream()
                                                .collect(Collectors.toMap(Person::getName, person -> person));
    
    System.out.println(nameToPersonMap);
    

    这段代码将根据 Person 对象的 name 属性将对象收集到一个 Map 中,键为 name,值为 Person 对象本身。

    3,自定义收集器

    除了使用预定义的收集器外,还可以通过实现 Collector 接口来创建自定义的收集器,以满足特定的需求。自定义收集器需要实现 Supplier, BiConsumer, BinaryOperator, Characteristics 接口方法。

    例如,假设我们要将流中的字符串连接成一个单独的字符串:

    List<String> letters = Arrays.asList("a", "b", "c", "d");
    
    String result = letters.stream()
                           .collect(StringBuilder::new, StringBuilder::append, StringBuilder::append)
                           .toString();
    
    System.out.println(result); 
    // 输出:abcd
    
    

    在这个例子中,我们通过 StringBuilder::new 提供了一个初始的累加器,然后使用 StringBuilder::append 将每个元素连接到累加器上,最后通过 StringBuilder::toString 获得最终的字符串结果。

    4,注意事项和使用场景

    在使用 collect 方法时,需要考虑以下几点:

    1. 并行流: 当使用并行流时,应确保收集器具备无状态(stateless)或线程安全(thread-safe)的特性,以避免并发问题。
    2. 性能考量: 使用合适的收集器能够有效地提升性能,特别是在处理大数据量时。
    3. 可变性: 尽量避免在并行流中使用可变的累加器,以免产生不可预料的结果。

    5,总结

    Stream 的 collect 方法是一个强大且灵活的工具,用于将流中的元素收集到不同类型的结果容器中。通过预定义的收集器或自定义的方式,可以满足各种不同的需求,使得集合操作变得简洁和高效。在实际开发中,充分利用 collect 方法能够大大简化代码,提高代码的可读性和维护性,是 Java 8 引入的函数式编程风格的重要体现之一。

  • 相关阅读:
    智能微秘书+FastGPT,打造你的超级微信助手!
    (历史上最详细的网络)华为初级网络工程师知识点总结(二)工作考研均受益
    通过JDBC连接MySQL
    LPC/LSP/LSF辨析
    Nwafu-OJ-1502 Problem 5 2019阶段1考试 题目3
    Python数值方法和可视化
    文件读写--python基础
    Http不转换成Https会有什么后果?
    【LC简单】387. 字符串中的第一个唯一字符
    技术分享 | my2sql 与 binlog2sql 解析效率 battle
  • 原文地址:https://blog.csdn.net/anyifeiyi/article/details/139987795