Bootstrap

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 引入的函数式编程风格的重要体现之一。

;