Я ищу краткий способ преобразовать 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);
}
Тем не менее, я получаю 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
но я не нашел ничего.
Stream.generate(iterator::next)
работает?Ответы:
Один из способов - создать Spliterator из Iterator и использовать его в качестве основы для вашего потока:
Альтернатива, которая может быть более читабельна, - это использовать Iterable - и создание Iterable из Iterator очень просто с помощью лямбды, потому что Iterable - это функциональный интерфейс:
источник
sourceIterator.next()
перед использованием потока, и вы увидите эффект (первый элемент не будет виден потоком).Iterable<String> iterable = () -> sourceIterator;
. Я должен признать, что мне потребовалось некоторое время, чтобы понять.Iterable<T>
этоFunctionalInterface
который имеет только один абстрактный методiterator()
. Как() -> sourceIterator
и лямбда-выражение, создающее экземплярIterable
экземпляра как анонимная реализация.() -> sourceIterator;
это сокращенная формаnew Iterable<>() { @Override public Iterator<String> iterator() { return sourceIterator; } }
Начиная с версии 21, библиотека Guava предоставляет
Streams.stream(iterator)
Он делает то , что @ assylias «ы ответ показывает .
источник
Отличное предложение! Вот мой подход к повторному использованию:
И использование (не забудьте статически импортировать asStream):
источник
Это возможно в Java 9.
источник
.parallel()
потоками. Они также выглядят немного медленнее, чем переходSpliterator
, даже для последовательного использования.parallel
может быть забавно, простота удивительна.Создание
Spliterator
сIterator
помощьюSpliterators
класса содержит более одной функции для создания spliterator, например , здесь я использую ,spliteratorUnknownSize
который получает итератор в качестве параметра, а затем создать поток с помощьюStreamSupport
источник
и использовать
Streams.stream(iterator)
:источник
Другой способ сделать это на Java 9+ с использованием Stream :: iterate (T, Predicate, UnaryOperator) :
источник
использование
Collections.list(iterator).stream()...
источник