O uso de lambdas e interfaces funcionais no Java 8 possibilita a criação de novas abstrações de loop. Eu posso repetir uma coleção com o índice e o tamanho da coleção:
List<String> strings = Arrays.asList("one", "two","three","four");
forEach(strings, (x, i, n) -> System.out.println("" + (i+1) + "/"+n+": " + x));
Quais saídas:
1/4: one
2/4: two
3/4: three
4/4: four
Que eu implementei como:
@FunctionalInterface
public interface LoopWithIndexAndSizeConsumer<T> {
void accept(T t, int i, int n);
}
public static <T> void forEach(Collection<T> collection,
LoopWithIndexAndSizeConsumer<T> consumer) {
int index = 0;
for (T object : collection){
consumer.accept(object, index++, collection.size());
}
}
As possibilidades são infinitas. Por exemplo, eu crio uma abstração que usa uma função especial apenas para o primeiro elemento:
forEachHeadTail(strings,
(head) -> System.out.print(head),
(tail) -> System.out.print(","+tail));
Que imprime uma lista separada por vírgula corretamente:
one,two,three,four
Que eu implementei como:
public static <T> void forEachHeadTail(Collection<T> collection,
Consumer<T> headFunc,
Consumer<T> tailFunc) {
int index = 0;
for (T object : collection){
if (index++ == 0){
headFunc.accept(object);
}
else{
tailFunc.accept(object);
}
}
}
As bibliotecas começarão a aparecer para fazer esse tipo de coisa, ou você pode criar suas próprias.
Type var = null; for (var : set) dosomething; if (var != null) then ...