[java] 반복자를 스트림으로 변환하는 방법?

나는 변환하는 간결한 방법을 찾고 있는데요 IteratorA를 Stream“보기”스트림으로 반복자 또는 더 구체적으로.

성능상의 이유로 새 목록에서 반복자의 사본을 피하고 싶습니다.

Iterator<String> sourceIterator = Arrays.asList("A", "B", "C").iterator();
Collection<String> copyList = new ArrayList<String>();
sourceIterator.forEachRemaining(copyList::add);
Stream<String> targetStream = copyList.stream();

의견의 몇 가지 제안을 바탕으로 다음과 같이 사용하려고했습니다 Stream.generate.

public static void main(String[] args) throws Exception {
    Iterator<String> sourceIterator = Arrays.asList("A", "B", "C").iterator();
    Stream<String> targetStream = Stream.generate(sourceIterator::next);
    targetStream.forEach(System.out::println);
}

그러나 NoSuchElementException(의 호출이 없기 때문에 hasNext)

Exception in thread "main" java.util.NoSuchElementException
    at java.util.AbstractList$Itr.next(AbstractList.java:364)
    at Main$$Lambda$1/1175962212.get(Unknown Source)
    at java.util.stream.StreamSpliterators$InfiniteSupplyingSpliterator$OfRef.tryAdvance(StreamSpliterators.java:1351)
    at java.util.Spliterator.forEachRemaining(Spliterator.java:326)
    at java.util.stream.ReferencePipeline$Head.forEach(ReferencePipeline.java:580)
    at Main.main(Main.java:20)

나는 살펴 보았다 StreamSupportCollections하지만 난 아무것도 찾지 못했습니다.



답변

한 가지 방법은 Iterator에서 Spliterator를 만들어 스트림의 기초로 사용하는 것입니다.

Iterator<String> sourceIterator = Arrays.asList("A", "B", "C").iterator();
Stream<String> targetStream = StreamSupport.stream(
          Spliterators.spliteratorUnknownSize(sourceIterator, Spliterator.ORDERED),
          false);

더 읽기 쉬운 대안은 Iterable을 사용하는 것입니다. Iterable은 기능적인 인터페이스이기 때문에 Iterator에서 Iterable을 만드는 것은 람다로 매우 쉽습니다.

Iterator<String> sourceIterator = Arrays.asList("A", "B", "C").iterator();

Iterable<String> iterable = () -> sourceIterator;
Stream<String> targetStream = StreamSupport.stream(iterable.spliterator(), false);


답변

버전 21부터 Guava 라이브러리는 Streams.stream(iterator)

@ assylias답변이 보여주는 것 입니다.


답변

좋은 제안! 재사용 가능한 테이크는 다음과 같습니다.

public class StreamUtils {

    public static <T> Stream<T> asStream(Iterator<T> sourceIterator) {
        return asStream(sourceIterator, false);
    }

    public static <T> Stream<T> asStream(Iterator<T> sourceIterator, boolean parallel) {
        Iterable<T> iterable = () -> sourceIterator;
        return StreamSupport.stream(iterable.spliterator(), parallel);
    }
}

그리고 사용법 (정적으로 asStream을 가져 오십시오) :

List<String> aPrefixedStrings = asStream(sourceIterator)
                .filter(t -> t.startsWith("A"))
                .collect(toList());


답변

이것은 Java 9에서 가능합니다.

Stream.generate(() -> null)
    .takeWhile(x -> iterator.hasNext())
    .map(n -> iterator.next())
    .forEach(System.out::println);


답변

만들기 Spliterator에서 Iterator사용 Spliterators여기에 사용하고, 예를 들어, 클래스가 spliterator를 만들기위한 하나 개 이상의 기능을 포함하고 spliteratorUnknownSize매개 변수로 반복자를 받고있는 다음 스트림 사용하여 작성StreamSupport

Spliterator<Model> spliterator = Spliterators.spliteratorUnknownSize(
        iterator, Spliterator.NONNULL);
Stream<Model> stream = StreamSupport.stream(spliterator, false);


답변

import com.google.common.collect.Streams;

그리고 사용 Streams.stream(iterator):

Streams.stream(iterator)
       .map(v-> function(v))
       .collect(Collectors.toList());


답변

사용하다 Collections.list(iterator).stream()...