
第一段引用上面的摘要:
本文介绍了在使用Java Stream API的collect()方法收集EnumSet时,如何正确提供Supplier。常见的错误是直接提供一个EnumSet实例,而不是一个生成EnumSet实例的Supplier。本文将详细讲解collect()方法的用法,并提供两种正确的实现方式,帮助开发者避免常见的编译错误。
Stream.collect()方法是Java Stream API中一个强大的终端操作,它允许我们将流中的元素收集到各种数据结构中,例如List、Set、Map等。它有多种重载形式,其中一种形式接受三个参数:
当我们需要收集到EnumSet时,常见的错误是直接将EnumSet.noneOf(ScConstraint.class)作为supplier参数传递,导致编译错误。这是因为supplier需要是一个Supplier接口的实现,而不是一个EnumSet实例。
正确的做法是使用Lambda表达式来创建一个Supplier,它会在每次调用时返回一个新的空的EnumSet。
import java.util.EnumSet;
import java.util.HashSet;
import java.util.Set;
import java.util.stream.Stream;
public class EnumSetCollector {
    public static void main(String[] args) {
        HashSet<Integer> nlsCandidates = new HashSet<>();
        nlsCandidates.add(1);
        nlsCandidates.add(2);
        nlsCandidates.add(3);
        EnumSet<ScConstraint>[] rowConstraints = new EnumSet[30];
        for (int i = 0; i < 30; i++) {
            rowConstraints[i] = EnumSet.allOf(ScConstraint.class);
        }
        Set<ScConstraint> s = EnumSet.of(ScConstraint.CONSTRAINT_A);
        int k = 0;
        EnumSet<ScConstraint> tbd = nlsCandidates.stream()
                .flatMap(p -> rowConstraints[10 * k + p].stream())
                .filter(cstr -> !s.contains(cstr))
                .collect(
                        () -> EnumSet.noneOf(ScConstraint.class), // Supplier
                        Set::add, // Accumulator
                        Set::addAll // Combiner
                );
        System.out.println(tbd);
    }
    enum ScConstraint {
        CONSTRAINT_A,
        CONSTRAINT_B,
        CONSTRAINT_C
    }
}在这个例子中,() -> EnumSet.noneOf(ScConstraint.class)就是一个Supplier,它会在每次调用时返回一个新的空的EnumSet<ScConstraint>。 Set::add 作为accumulator,将流中的每个元素添加到EnumSet中。Set::addAll 作为combiner,合并两个EnumSet。
另一种更简洁的方式是使用Collectors.toCollection(),它接受一个Supplier作为参数,并返回一个Collector。
import java.util.EnumSet;
import java.util.HashSet;
import java.util.Set;
import java.util.stream.Collectors;
public class EnumSetCollector {
    public static void main(String[] args) {
        HashSet<Integer> nlsCandidates = new HashSet<>();
        nlsCandidates.add(1);
        nlsCandidates.add(2);
        nlsCandidates.add(3);
        EnumSet<ScConstraint>[] rowConstraints = new EnumSet[30];
        for (int i = 0; i < 30; i++) {
            rowConstraints[i] = EnumSet.allOf(ScConstraint.class);
        }
        Set<ScConstraint> s = EnumSet.of(ScConstraint.CONSTRAINT_A);
        int k = 0;
        EnumSet<ScConstraint> tbd = nlsCandidates.stream()
                .flatMap(p -> rowConstraints[10 * k + p].stream())
                .filter(cstr -> !s.contains(cstr))
                .collect(Collectors.toCollection(
                        () -> EnumSet.noneOf(ScConstraint.class)
                ));
        System.out.println(tbd);
    }
    enum ScConstraint {
        CONSTRAINT_A,
        CONSTRAINT_B,
        CONSTRAINT_C
    }
}这种方式更加简洁,可读性也更好。 Collectors.toCollection() 内部已经处理了accumulator和combiner的逻辑,只需要提供一个创建EnumSet的supplier即可。
在使用Stream.collect()方法收集EnumSet时,正确提供Supplier是关键。避免直接提供EnumSet实例,而是使用Lambda表达式或Collectors.toCollection()来创建Supplier。理解collect()方法的各个参数的含义,可以帮助我们更好地利用Stream API,编写出高效、简洁的代码。
以上就是使用Stream.collect()正确收集EnumSet的详细内容,更多请关注php中文网其它相关文章!
 
                        
                        每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。
 
                Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号