Потоки обеспечивают намного лучшую абстракцию для составления различных операций, которые вы хотите выполнить над наборами или потоками поступающих данных. Особенно, когда вам нужно отобразить элементы, отфильтровать и преобразовать их.
Ваш пример не очень практичен. Рассмотрим следующий код с сайта Oracle .
List<Transaction> groceryTransactions = new Arraylist<>();
for(Transaction t: transactions){
if(t.getType() == Transaction.GROCERY){
groceryTransactions.add(t);
}
}
Collections.sort(groceryTransactions, new Comparator(){
public int compare(Transaction t1, Transaction t2){
return t2.getValue().compareTo(t1.getValue());
}
});
List<Integer> transactionIds = new ArrayList<>();
for(Transaction t: groceryTransactions){
transactionsIds.add(t.getId());
}
может быть написано с использованием потоков:
List<Integer> transactionsIds =
transactions.stream()
.filter(t -> t.getType() == Transaction.GROCERY)
.sorted(comparing(Transaction::getValue).reversed())
.map(Transaction::getId)
.collect(toList());
Второй вариант гораздо более читабелен. Поэтому, когда у вас есть вложенные циклы или различные циклы, выполняющие частичную обработку, это очень хороший кандидат для использования Streams / Lambda API.