«1. Обзор

Java Stream API был главной особенностью версии Java 8. Потоки представляют собой последовательности объектов с ленивой оценкой и предоставляют богатый, плавный и похожий на монадический API.

В этой статье мы кратко рассмотрим способы получения последнего элемента потока. Имейте в виду, что из-за природы потоков это не естественная операция. Всегда убедитесь, что вы не работаете с бесконечными потоками.

2. Использование API Reduce

Reduce, проще говоря, сокращает набор элементов в потоке до одного элемента.

В этом случае мы уменьшим набор элементов, чтобы получить последний элемент в потоке. Имейте в виду, что этот метод будет возвращать детерминированные результаты только для последовательных потоков.

Давайте воспользуемся списком строковых значений, получим поток из списка и затем уменьшим:

List<String> valueList = new ArrayList<>();
valueList.add("Joe");
valueList.add("John");
valueList.add("Sean");

Stream<String> stream = valueList.stream();
stream.reduce((first, second) -> second)
  .orElse(null);

Здесь поток урезан до уровня, на котором остается только последний элемент. Если поток пуст, он вернет нулевое значение.

2. Использование функции пропуска

Другой способ получить последний элемент потока — пропустить все элементы перед ним. Этого можно добиться с помощью функции Skip класса Stream. Имейте в виду, что в этом случае мы потребляем поток дважды, поэтому есть явное влияние на производительность.

Давайте создадим список строковых значений и используем его функцию размера, чтобы определить, сколько элементов нужно пропустить, чтобы достичь последнего элемента.

Вот пример кода, получающего последний элемент с помощью пропуска:

List<String> valueList = new ArrayList<String>();
valueList.add("Joe");
valueList.add("John");
valueList.add("Sean");

long count = valueList.stream().count();
Stream<String> stream = valueList.stream();
   
stream.skip(count - 1).findFirst().get();

«Шон» оказывается последним элементом.

4. Получение последнего элемента бесконечного потока

Попытка получить последний элемент бесконечного потока привела бы к бесконечной последовательности вычислений, выполняемых над бесконечными элементами. Как skip, так и reduce не вернутся из выполнения оценки, если мы не ограничим бесконечный поток определенным количеством элементов, используя операцию ограничения.

Вот пример кода, в котором мы взяли бесконечный поток и попытались получить последний элемент:

Stream<Integer> stream = Stream.iterate(0, i -> i + 1);
stream.reduce((first, second) -> second).orElse(null);

Следовательно, поток не вернется из оценки и в конечном итоге остановит выполнение программы. .

5. Заключение

Мы видели разные способы получения последнего элемента потока как с помощью API сокращения, так и с помощью Skip. Мы также рассмотрели, почему это невозможно с бесконечными потоками.

Мы видели, что получить последний элемент из Stream не так просто, как получить его из других структур данных. Это связано с ленивым характером потоков, которые не оцениваются, пока не будет вызвана терминальная функция, и мы никогда не узнаем, является ли текущий оцениваемый элемент последним.

Фрагменты кода можно найти на GitHub.