Program Tip

Java 8 스트림을 Guava ImmutableCollection으로 수집하려면 어떻게해야합니까?

programtip 2020. 10. 23. 08:21
반응형

Java 8 스트림을 Guava ImmutableCollection으로 수집하려면 어떻게해야합니까?


다음을 수행하고 싶습니다.

List<Integer> list = IntStream.range(0, 7).collect(Collectors.toList());

그러나 결과 목록은 Guava의 ImmutableList.

내가 할 수 있다는 걸 알아

List<Integer> list = IntStream.range(0, 7).collect(Collectors.toList());
List<Integer> immutableList = ImmutableList.copyOf(list);

하지만 직접 수집하고 싶습니다. 난 노력 했어

List<Integer> list = IntStream.range(0, 7)
    .collect(Collectors.toCollection(ImmutableList::of));

하지만 예외가 발생했습니다.

com.google.common.collect.ImmutableCollection.add (ImmutableCollection.java:96)의 java.lang.UnsupportedOperationException


toImmutableList()렉시의 허용 대답에있어서 현재 포함되어 구아바 (21) 와 같이 사용할 수있다 :

ImmutableList<Integer> list = IntStream.range(0, 7)
    .boxed()
    .collect(ImmutableList.toImmutableList());

collectingAndThen수집기가 유용한 곳은 다음과 같습니다.

List<Integer> list = IntStream.range(0, 7).boxed()
                .collect(collectingAndThen(toList(), ImmutableList::copyOf));

List방금 구축 한 변형을 적용합니다 . 결과는 ImmutableList.


또는에 직접 모아서 마지막에 Builder전화 할 수 있습니다 build().

List<Integer> list = IntStream.range(0, 7)
                .collect(Builder<Integer>::new, Builder<Integer>::add, (builder1, builder2) -> builder1.addAll(builder2.build()))
                .build();

이 옵션이 다소 장황하고 여러 곳에서 사용하려는 경우 고유 한 수집기를 만들 수 있습니다.

class ImmutableListCollector<T> implements Collector<T, Builder<T>, ImmutableList<T>> {
    @Override
    public Supplier<Builder<T>> supplier() {
        return Builder::new;
    }

    @Override
    public BiConsumer<Builder<T>, T> accumulator() {
        return (b, e) -> b.add(e);
    }

    @Override
    public BinaryOperator<Builder<T>> combiner() {
        return (b1, b2) -> b1.addAll(b2.build());
    }

    @Override
    public Function<Builder<T>, ImmutableList<T>> finisher() {
        return Builder::build;
    }

    @Override
    public Set<Characteristics> characteristics() {
        return ImmutableSet.of();
    }
}

그리고:

List<Integer> list = IntStream.range(0, 7)
                              .boxed()
                              .collect(new ImmutableListCollector<>());

댓글에서 링크가 사라지는 경우를 대비하여 내 두 번째 접근 방식은 단순히 사용하는 정적 유틸리티 메서드에서 정의 할 수 있습니다 Collector.of. 자신 만의 Collector수업을 만드는 것보다 간단합니다 .

public static <T> Collector<T, Builder<T>, ImmutableList<T>> toImmutableList() {
    return Collector.of(Builder<T>::new, Builder<T>::add, (l, r) -> l.addAll(r.build()), Builder<T>::build);
}

및 사용법 :

 List<Integer> list = IntStream.range(0, 7)
                               .boxed()
                               .collect(toImmutableList());

내 질문에 대한 직접적인 대답은 아니지만 (컬렉터를 사용하지 않음) 중간 컬렉션을 사용하지 않는 상당히 우아한 접근 방식입니다.

Stream<Integer> stream = IntStream.range(0, 7).boxed();
List<Integer> list = ImmutableList.copyOf(stream.iterator());

소스 .


BTW : JDK 10 이후 순수 Java로 수행 할 수 있습니다.

List<Integer> list = IntStream.range(0, 7)
    .collect(Collectors.toUnmodifiableList());

또한 toUnmodifiableSettoUnmodifiableMap가능합니다.

내부 수집기는 다음을 통해 수행되었습니다. List.of(list.toArray())


참고로 Java 8없이 Guava에서이 작업을 수행하는 합리적인 방법이 있습니다.

ImmutableSortedSet<Integer> set = ContiguousSet.create(
    Range.closedOpen(0, 7), DiscreteDomain.integers());
ImmutableList<Integer> list = set.asList();

If you don't actually need the List semantics and can just use a NavigableSet, that's even better since a ContiguousSet doesn't have to actually store all the elements in it (just the Range and DiscreteDomain).

참고URL : https://stackoverflow.com/questions/29013250/how-can-i-collect-a-java-8-stream-into-a-guava-immutablecollection

반응형