IDEA предлагает заменить, например, следующее:
for (Point2D vertex : graph.vertexSet()) {
union.addVertex(vertex);
}
с этим:
graph.vertexSet().forEach(union::addVertex);
Эта новая версия более читаема. Но есть ли ситуации, когда я лучше придерживаюсь старой конструкции старого языка для итераций вместо использования нового метода foreach
?
Например, если я правильно понимаю, механизм ссылок метода подразумевает построение анонимного объекта Consumer
, который в противном случае (с конструкцией языка for
) не будет создан. Может ли это стать узким местом для некоторых действий?
Итак, я написал этот не очень исчерпывающий тест:
package org.sample;
import org.openjdk.jmh.annotations.Benchmark;
import org.openjdk.jmh.annotations.Fork;
import org.openjdk.jmh.annotations.Threads;
import org.openjdk.jmh.infra.Blackhole;
import org.tendiwa.geometry.Point2D;
import java.util.ArrayList;
import java.util.List;
import java.util.stream.Collectors;
import java.util.stream.IntStream;
public class LanguageConstructVsForeach {
private static final int NUMBER_OF_POINTS = 10000;
private static final List<Point2D> points = IntStream
.range(0, NUMBER_OF_POINTS)
.mapToObj(i -> new Point2D(i, i * 2))
.collect(Collectors.toList());
@Benchmark
@Threads(1)
@Fork(3)
public void languageConstructToBlackhole(Blackhole bh) {
for (Point2D point : points) {
bh.consume(point);
}
}
@Benchmark
@Threads(1)
@Fork(3)
public void foreachToBlackhole(Blackhole bh) {
points.forEach(bh::consume);
}
@Benchmark
@Threads(1)
@Fork(3)
public List<Point2D> languageConstructToList(Blackhole bh) {
List<Point2D> list = new ArrayList<>(NUMBER_OF_POINTS);
for (Point2D point : points) {
list.add(point);
}
return list;
}
@Benchmark
@Threads(1)
@Fork(3)
public List<Point2D> foreachToList(Blackhole bh) {
List<Point2D> list = new ArrayList<>(NUMBER_OF_POINTS);
points.forEach(list::add);
return list;
}
}
И получил:
Benchmark Mode Samples Score Error Units
o.s.LanguageConstructVsForeach.foreachToBlackhole thrpt 60 33693.834 ± 894.138 ops/s
o.s.LanguageConstructVsForeach.foreachToList thrpt 60 7753.941 ± 239.081 ops/s
o.s.LanguageConstructVsForeach.languageConstructToBlackhole thrpt 60 16043.548 ± 644.432 ops/s
o.s.LanguageConstructVsForeach.languageConstructToList thrpt 60 6499.527 ± 202.589 ops/s
Как получается foreach
более эффективен в обоих случаях: когда я фактически ничего не делаю и когда делаю какую-то фактическую работу? Не foreach
просто инкапсулировать Iterator
? Является ли этот показатель даже правильным? Если это так, есть ли какая-нибудь причина сегодня использовать конструкцию старого языка с Java 8?