Разница в производительности между лямбдами Java 8 и анонимными внутренними классами
До Java 8 функциональность лямбды могла быть достигнута с помощью анонимных внутренних классов. Например:
interface Lambda {
void doStuff();
}
// ...
public void doWithCallback(Lambda callback) {
// ...
callback.doStuff();
}
// ...
doWithCallback(new Lambda {
public void doStuff() {
// ...
}
});
С точки зрения производительности, есть ли разница между тем, чтобы продолжать использовать этот подход и использовать новые лямбды Java 8?
2 ответа
Oracle опубликовал исследование, сравнивающее производительность между Lambdas и анонимными классами
См. JDK 8: Лямбда-исследование, выполненное Сергеем Куксенко, длиной 74 слайда.
Резюме: медленно разогревается, но когда JIT указывает на это, это наихудший случай так же быстро, как анонимный класс, но может быть быстрее.
Как я обнаружил, перебор массива с Stream работает намного медленнее (74 слайда не учитывают такой случай). Я думаю, что это не единственная утечка производительности в лямбдах (думаю, это будет улучшено в будущем). Пример ниже работал с Java 8 без каких-либо опций:
//Language is an enum
Language[] array = Language.values();
System.err.println(array.length); // 72 items
long t = System.nanoTime();
for (Language l : array) System.out.println(l.getLanguageName());
System.err.println(System.nanoTime()-t); //nano time 1864724
t = System.nanoTime();
Arrays.stream(array).forEach(v -> System.out.println(v.getLanguageName()));
System.err.println(System.nanoTime()-t); //nano time 55812625 (55812625/1864724 = 29.93 times longer)
List<Language> list = Arrays.asList(array);
t = System.nanoTime();
for (Language l : list) System.out.println(l.getLanguageName());
System.err.println(System.nanoTime()-t); //nano time 1435008
t = System.nanoTime();
list.forEach(v -> System.out.println(v.getLanguageName()));
System.err.println(System.nanoTime()-t); //nano time 1619973 (1619973/1435008 = 1.128 times longer)