Как конвертировать итератор в поток?
Я ищу сжатый способ преобразования Iterator
в 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);
}
Тем не менее, я получаю a 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)
Я посмотрел StreamSupport
и Collections
но я ничего не нашел.
Ответы
Ответ 1
Один из способов - создать Spliterator из Iterator и использовать его в качестве основы для вашего потока:
Iterator<String> sourceIterator = Arrays.asList("A", "B", "C").iterator();
Stream<String> targetStream = StreamSupport.stream(
Spliterators.spliteratorUnknownSize(sourceIterator, Spliterator.ORDERED),
false);
Альтернативой, которая может быть более читаемой, является использование Iterable - и создание Iterable из Iterator очень просто с lambdas, потому что Iterable - это функциональный интерфейс:
Iterator<String> sourceIterator = Arrays.asList("A", "B", "C").iterator();
Iterable<String> iterable = () -> sourceIterator;
Stream<String> targetStream = StreamSupport.stream(iterable.spliterator(), false);
Ответ 2
Начиная с версии 21, библиотека Guava предоставляет Streams.stream(iterator)
Он делает то, что показывает ответ @assylias.
Ответ 3
Отличное предложение!
Здесь мое повторное использование:
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());
Ответ 4
Это возможно в java 9, используя только потоки, но есть предостережение, на которое стоит обратить внимание. В следующем наивном примере не удастся распечатать последний элемент итератора.
Stream.generate(iterator::next)
.takeWhile(i -> iterator.hasNext())
.forEach(System.out::println);
Это, однако, работает правильно:
Stream.generate(() -> null)
.takeWhile(x -> iterator.hasNext())
.map(n -> iterator.next())
.forEach(System.out::println);
Ответ 5
Создать Spliterator
из Iterator
, используя класс Spliterators
, содержит более одной функции для создания spliterator, например, здесь я использую spliteratorUnknownSize
, который получает итератор в качестве параметра, а затем создает поток с использованием StreamSupport
Spliterator<Model> spliterator = Spliterators.spliteratorUnknownSize(
iterator, Spliterator.NONNULL);
Stream<Model> stream = StreamSupport.stream(spliterator, false);
Ответ 6
Вы можете просто использовать функцию ограничения с помощью итератора:
Stream.generate(iterator::next).limit(list.size())
Ответ 7
public static <T> Stream<T> stream(Collection<T> c) {
return StreamSupport.stream(((Iterable<T>)()->c.iterator()).spliterator(), false);
}
Ответ 8
Используйте Collections.list(iterator).stream()...