Эта заметка является писанной версией моего доклада "Как загубить производительность с помощью неэффективного кода" с конференции JPoint 2018. Посмотреть видео и слайды можно на странице конференции. В расписании доклад отмечен обидным стаканчиком смузи, так что ничего сверхсложного не будет, это скорее для начинающих.


Предмет доклада:


  • как смотреть на код, чтобы найти в нём узкие места
  • распространённые антипаттерны
  • неочевидные грабли
  • обход граблей

В кулуарах мне указали на некоторые неточности/упущения в докладе, они здесь отмечены. Замечания также приветствуются.


Влияние порядка исполнения на производительность


Есть класс пользователя:


class User {
  String name;
  int age;
}

Нам нужно сравнивать объекты между собой, поэтому объявим методы equals и hashCode:


import lombok.EqualsAndHashCode;

@EqualsAndHashCode
class User {
  String name;
  int age;
}

Код работоспособен, вопрос в другом: будет ли производительность данного кода наилучшей? Чтобы ответить на него давайте вспомним об особенности метода Object::equals: он возвращает положительный результат лишь тогда, когда все сравниваемые поля равны, в противном же случае результат будет отрицательным. Иными словами, одного отличия уже достаточно для отрицательного результата.


Посмотрев код, создаваемый для @EqualsAndHashCode увидим примерно вот это:


public boolean equals(Object that) {
  //...
  if (name == null && that.name != null) {
    return false;
  }
  if (name != null && !name.equals(that.name)) {
    return false;
  }
  return age == that.age;
}

Порядок проверки полей соответствует порядку их объявления, что в нашем случае не лучшее решение, ведь сравнение объектов с помощью equals "тяжелее" сравнения простых типов.


Хорошо, попробуем создать методы equals/hashCode с помощью "Идеи":


@Override
public boolean equals(Object o) {
  if (this == o) {
    return true;
  }
  if (o == null || getClass() != o.getClass()) {
    return false;
  }
  User that = (User) o;
  return age == that.age && Objects.equals(name, that.name);
}

"Идея" создаёт более умный код, который знает о сложности сравнения разных видов данных. Ну хорошо, мы выбросим @EqualsAndHashCode и будем явно прописывать equals/hashCode. Теперь посмотрим, что происходит при расширении класса:


class User {
  List<T> props;
  String name;
  int age;
}

Пересоздадим equals/hashCode:


@Override
public boolean equals(Object o) {
  if (this == o) {
    return true;
  }
  if (o == null || getClass() != o.getClass()) {
    return false;
  }
  User that = (User) o;
  return age == that.age
      && Objects.equals(props, that.props)   // <----
      && Objects.equals(name, that.name);
}

Сравнение списков выполняется до сравнения строк, что бессмысленно, когда строки разные. На первый взгляд, особой разницы нет, ведь строки равной длины сравниваются познаково (т. е. время сравнение растёт вместе с длиной строки):


Здесь была неточность

Метод java.lang.String::equals интринзифицирован, поэтому познакового сравнения при исполнении не происходит.


//java.lang.String
public boolean equals(Object anObject) {
  if (this == anObject) {
    return true;
  }
  if (anObject instanceof String) {
    String anotherString = (String) anObject;
    int n = value.length;
    if (n == anotherString.value.length) {
      char v1[] = value;
      char v2[] = anotherString.value;
      int i = 0;
      while (n-- != 0) {     // <----
        if (v1[i] != v2[i])
          return false;
        i++;
      }
      return true;
    }
  }
  return false;
}

Теперь рассмотрим сравнение двух ArrayList-ов (как наиболее часто используемой реализации списка). Исследовав ArrayList с удивлением обнаружим, что собственной реализации equals у него нет, а используется унаследованная реализация:



//AbstractList::equals

public boolean equals(Object o) {
  if (o == this) {
    return true;
  }
  if (!(o instanceof List)) {
    return false;
  }

  ListIterator<E> e1 = listIterator();
  ListIterator<?> e2 = ((List<?>) o).listIterator();
  while (e1.hasNext() && e2.hasNext()) {    // <----
    E o1 = e1.next();
    Object o2 = e2.next();
    if (!(o1 == null ? o2 == null : o1.equals(o2))) {
      return false;
    }
  }
  return !(e1.hasNext() || e2.hasNext());
}

Важным здесь является создание двух итераторов и попарный проход по ним. Предположим, есть два ArrayList-а:


  • в одном числа от 1 до 99
  • во втором числа от 1 до 100

В идеале было бы достаточно сравнить размеры двух списков и при несовпадении сразу вернуть отрицательный результат (как это делает AbstractSet), в действительности же выполнится 99 сравнений и лишь на сотом станет понятно, что списки отличаются.


Чё там у котлиновцев?


data class User(val name: String, val age: Int);

Тут всё как у ломбока — порядок сравнения соответствует порядку объявления:


public boolean equals(Object o) {
  if (this == o) {
    return true;
  }
  if (o instanceof User) {
    User u = (User) o;
    if (Intrinsics.areEqual(name, u.name) && age == u.age) {  // <----
      return true;
    }
  }
  return false;
}

Как улучшить положение? - Жаловаться!

IDEA-170178 Comparison of collections should be executed at the very end of IDEA-generated equals()
https://youtrack.jetbrains.com/issue/IDEA-170178


Reordering of comparison in @EqualsAndHashCode for better performance #1543
https://github.com/rzwitserloot/lombok/issues/1543


KT-23184 Auto-generated equals() of data class is not optimal
https://youtrack.jetbrains.com/issue/KT-23184


В качестве обхода можно вручную упорядочивать объявления полей.


Усложним задачу


void check(Dto dto) {
  SomeEntity entity = jpaRepository.findOne(dto.getId());
  boolean valid = dto.isValid();

  if (valid && entity.hasGoodRating()) { // <----
    //do smth
  }
}

Код предполагает обращение к БД даже тогда, когда итог проверки отмеченного стрелкой условия заранее предсказуем. Если значение переменной valid ложно, то код в блоке if никогда не выполнится, а значит можно обойтись без запроса:


void check(Dto dto) {
  boolean valid = dto.isValid();
  if (valid && hasGoodRating(dto)) {
    //do smth
  }
}

// отложенное исполнение можно построить и на предикате, это дело вкуса
boolean hasGoodRating(Dto dto) {
  SomeEntity entity = jpaRepository.findOne(dto.getId());
  return entity.hasGoodRating();
}

Замечание из кулуаров

Проседание может быть незначительным, когда возвращаемая из JpaRepository::findOne сущность уже находится в кэше первого уровня, — тогда запроса не будет.


Похожий пример без явного ветвления:


boolean checkChild(Dto dto) {
  Long id = dto.getId();
  Entity entity = jpaRepository.findOne(id);
  return dto.isValid() && entity.hasChild();
}

Быстрый возврат позволяет отложить запрос:


boolean checkChild(Dto dto) {
  if (!dto.isValid()) {
    return false;
  }
  return jpaRepository.findOne(dto.getId()).hasChild();
}

Достаточно очевидное дополнение, не попавшее в доклад

Представьте, что некая проверка использует подобную сущность:


@Entity
class ParentEntity {
  @ManyToOne(fetch = LAZY)
  @JoinColumn(name = "CHILD_ID")
  private ChildEntity child;

  @Enumerated(EnumType.String)
  private SomeType type;

Если проверка использует одну и ту же сущность, то стоит позаботится о том, чтобы обращение к "ленивым" дочерним сущностям/коллекциям выполнялось после обращения к полям, которые уже загружены. На первый взгляд один дополнительный запрос не окажет значительного влияния на общую картину, но всё может изменится при выполнении действия в цикле.


Вывод: цепочки действий/проверок стоит упорядочивать в порядке возрастания сложности отдельных операций, возможно часть из них выполнять не придётся.


Циклы и массовая обработка


Следующий пример в особых пояснениях не нуждается:


@Transactional
void enrollStudents(Set<Long> ids) {
  for (Long id : ids) {
    Student student = jpaRepository.findOne(id); // <---- O(n)
    enroll(student);
  }
}

Из-за множественных запросов к БД код работает медленно.


Замечание

Производительность может просесть ещё сильнее, если метод enrollStudents исполняется вне транзакции: тогда каждый вызов o.s.d.j.r.JpaRepository::findOne будет выполнятся в новой транзакции (см. SimpleJpaRepository), что означает получение и возврат соединения с БД, а также создание и сброс кэша первого уровня.


Исправим:


@Transactional
void enrollStudents(Set<Long> ids) {
  if (ids.isEmpty()) {
    return;
  }
  for (Student student : jpaRepository.findAll(ids)) {
    enroll(student);
  }
}

Измерим время выполнения (в мкс) для коллекции ключей (10 и 100 штук)


Бенчмарк


Замечание

Если вы используете Oracle и передаёте более 1000 ключей в findAll, то вы получите исключение ORA-01795: maximum number of expressions in a list is 1000.
Также выполнения тяжелого (со множеством ключей) in-запроса может оказаться хуже, чем n запросов. Тут всё сильно зависит от конкретного приложения, поэтому механическое замена цикла на массовую обработку может ухудшить производительность.


Более сложный пример на ту же тему


for (Long id : ids) {
  Region region = jpaRepository.findOne(id);
  if (region == null) {        // <---- затруднение
    region = new Region();
    region.setId(id);
  }
  use(region);
}

В данном случае мы не можем заменить цикл на JpaRepository::findAll, т. к. это сломает логику: все значения полученные из JpaRepository::findAll будут не null и блок if не отработает.


Разрешить это затруднение нам поможет то обстоятельство, что для каждого ключа БД
возвращает либо действительное значение, либо его отсутствие. Т. е. в некотором смысле БД — это словарь. Ява из коробки даёт нам готовую реализацию словаря — HashMap — поверх которой мы и построим логику подмены БД:


Map<Long, Region> regionMap = jpaRepository.findAll(ids)
  .stream()
  .collect(Collectors.toMap(Region::getId, Function.identity()));

for (Long id : ids) {
  Region region = map.get(id);
  if (region == null) {
    region = new Region();
    region.setId(id);
  }
  use(region);
}

Пример в обратную сторону


// class Saver
@Transactional(propagation = Propagation.REQUIRES_NEW)
public void save(List<AuditEntity> entities) {
  jpaRepository.save(entities);
}

Этот код всегда создаёт новую транзакцию для сохранения списка сущностей. Проседание начинается при множественных вызовах метода, открывающего новую транзакцию:


// другой класс
@Transactional
public void audit(List<AuditDto> inserts) {
  inserts.map(this::toEntities).forEach(saver::save);  // <----
}

// class Saver

@Transactional(propagation = Propagation.REQUIRES_NEW) // <----
public void save(List<AuditEntity> entities) {
  jpaRepository.save(entities);
}

Решение: применять метод Saver::save сразу для всего набора данных:


@Transactional
public void audit(List<AuditDto> inserts) {
  List<AuditEntity> bulk = inserts
    .map(this::toEntities)
    .flatMap(List::stream)  // <----
    .collect(toList());
  saver.save(bulk);
}

Множество транзакций сливается в одну, что даёт ощутимый прирост (время в мкс):


Бенчмарк


Пример со множественными транзакциями сложно формализовать, чего не скажешь о вызове JpaRepository::findOne в цикле.


Поэтому примем меры

https://youtrack.jetbrains.com/issue/IDEA-165730
IDEA-165730 Warn about inefficient usage of JpaRepository


https://youtrack.jetbrains.com/issue/IDEA-165942
IDEA-165942 Inspection to replace method call in a loop with bulk operation
Исправлено в 2017.1


Подход применим не только к БД, поэтому Тагир lany Валеев пошел дальше. И если раньше мы писали вот так:


List<Long> list = new ArrayList<>();
for (Long id : items) {
  list.add(id);
}

и всё было хорошо, то теперь "Идея" предлагает исправиться:


List<Long> list = new ArrayList<>();
list.addAll(items);

Но даже этот вариант удовлетворяет её не всегда, ведь можно сделать ещё короче и быстрее:


List<Long> list = new ArrayList<>(items);

Сравним (время в нс)

Для ArrayList-а это улучшение даёт заметный прирост:



Для HashSet-а всё не так радужно:



Бенчмарк


Полезное для разработчика:

https://youtrack.jetbrains.com/issue/IDEA-138456
IDEA-138456 New inspection: Collection.addAll() replaceable with parametrized constructor
Исправлено в 142.1217


https://youtrack.jetbrains.com/issue/IDEA-178761
IDEA-178761 Inspection 'Collection.addAll() replaceable with parametrized constructor' should be turned on by default
Исправлено в 2017.3


Удаление из ArrayList


for (int i = from; i < to; i++) {
  list.remove(from);
}

Проблема в реализации метода List::remove:


public E remove(int index) {
  Objects.checkIndex(index, size);
  modCount++;
  E oldValue = elementData(index);
  int numMoved = size - index - 1;
  if (numMoved > 0) {
    System.arraycopy(array, index + 1, array, index, numMoved); // <----
  }
  array[--size] = null; // clear to let GC do its work
  return oldValue;
}

Решение:


list.subList(from, to).clear();

Но что если в исходном коде удалённое значение используется?


for (int i = from; i < to; i++) {
  E removed = list.remove(from);
  use(removed);
}

Теперь нужно предварительно пройти по очищаемому подсписку:


List<String> removed = list.subList(from, to);
removed.forEach(this::use);
removed.clear();

Если же очень хочется удалять в цикле, то облегчить боль поможет смена направления прохода по списку. Смысл его в том, чтобы сдвигать меньшее количество элементов после очистки ячейки:


//прямой проход менее эффективен, т. к. сдвигаются все элементы справа от удаляемого
for (int i = from; i < to; i++) {
  E removed = list.remove(from);
  use(removed, i);
}

//обратный проход быстрее, т. к. сдвиг всегда постоянный
for (int i = to - 1; i >= from; i--) {
  E removed = list.remove(i);
  use(removed, reverseIndex(i));
}

Сравним все три способа (под столбцами указан % удаляемых элементов из списка размером 100):


Java 8


Java 9


Кстати, кто-то заметил аномалию?


Узреть


Если мы удаляем половину всех данных, двигаясь с конца, то удаляется всегда последний элемент и сдвига не происходит:


// ArrayList
public E remove(int index) {
  rangeCheck(index);

  modCount++;
  E oldValue = elementData(index);

  int numMoved = size - index - 1;
  if (numMoved > 0) {  // <---- условие никогда не выполняется
    System.arraycopy(elementData, index+1, elementData, index, numMoved);
  }
  elementData[--size] = null; // clear to let GC do its work

  return oldValue;
}

Бенчмарк


Полезное для разработчика

https://youtrack.jetbrains.com/issue/IDEA-177466
IDEA-177466 Detect List.remove(index) called in a loop
Исправлено в 2018.2


Вывод: массовые операции часто быстрее одиночных.


Область видимости и производительность


Этот код в особых пояснениях не нуждается:


void leaveForTheSecondYear() {
  List<Student> naughty = repository.findNaughty();
  List<Student> underAchieving = repository.findUnderAchieving(); // <----

  if (settings.leaveBothCategories()) {
    leaveForTheSecondYear(naughty, underAchieving); // <----
    return;
  }
  leaveForTheSecondYear(naughty);
}

Сужаем область видимости, что даёт минус 1 запрос:


void leaveForTheSecondYear() {
  List<Student> naughty = repository.findNaughty();

  if (Settings.leaveBothCategories()) {  
    List<Student> underAchieving = repository.findUnderAchieving(); // <----
    leaveForTheSecondYear(naughty, underAchieving); // <----
    return;
  }
  leaveForTheSecondYear(naughty);
}

И тут внимательный читатель должен спросить: а как же статический анализ? Почему "Идея" нам не подсказала о лежащем на поверхности улучшении?


Дело в том, что возможности статического анализа ограничены: если метод сложный (тем более взаимодействующий с БД) и влияет на общее состояние, то перенос его исполнения может сломать приложение. Статический анализатор в состоянии сообщить об очень простых исполнениях, перенос которых, скажем, внутрь блока ничего не сломает.


В качестве эрзаца можно использовать подсветку переменных, но, повторюсь, использовать осторожно, т. к. всегда возможны побочные эффекты. Можно использовать аннотацию @org.jetbrains.annotations.Contract(pure = true), доступную из библиотеки jetbrains-annotations для обозначения методов, не изменяющих состояние:


// com.intellij.util.ArrayUtil
@Contract(pure = true)
public static int find(@NotNull int[] src, int obj) {
  return indexOf(src, obj);
}

Вывод: чаще всего лишняя работа только ухудшает производительность.


Самый необычный пример


@Service
public class RemoteService {
  private ContractCounter contractCounter;

  @Transactional(readOnly = true)   // <----
  public int countContracts(Dto dto) {
    if (dto.isInvalid()) {
      return -1;   // <----
    }
    return contractCounter.countContracts(dto);
  }
}

Эта реализации открывает транзакцию даже в том случае, когда транзакционность не нужна (быстрый возврат -1 из метода).


Всё что нужно сделать, это убрать транзакционность внутрь метода ContractCounter::countContracts, где она необходима, а из "внешнего" метода её убрать.


Сравним время исполнения для случая, когда возвращается -1 (нс):


Сравним потребление памяти (байт):


Бенчмарк


Вывод: контроллеры и смотрящие "наружу" сервисы нужно освобождать от транзакционности (это не их ответственность) и выносить туда всю логику проверки входных данных, не требующую обращения к БД и транзакционным компонентам.


Преобразование даты/времени в строку


Одна из вечных задач — превращение даты/времени в строку. До "восьмёрки" мы делали так:


SimpleDateFormat formatter = new SimpleDateFormat("dd.MM.yyyy");
String dateAsStr = formatter.format(date);

С выходом JDK 8 у нас появился LocalDate/LocalDateTime и, соответственно, DateTimeFormatter


DateTimeFormatter formatter = ofPattern("dd.MM.yyyy");
String dateAsStr = formatter.format(localDate);

Измерим его производительность:


Date date = new Date();
LocalDate localDate = LocalDate.now();
SimpleDateFormat sdf = new SimpleDateFormat("dd.MM.yyyy");
DateTimeFormatter dtf = DateTimeFormatter.ofPattern("dd.MM.yyyy");

@Benchmark
public String simpleDateFormat() {
  return sdf.format(date);
}

@Benchmark
public String dateTimeFormatter() {
  return dtf.format(localDate);
}

Время (нс):


Память (байт):


Вопрос: допустим наш сервис получает данные извне и мы не можем отказаться от java.util.Date. Будет ли выгодно нам преобразовать Date в LocalDate, если последний быстрее преобразовывается в строку? Посчитаем:


@Benchmark
public String measureDateConverted(Data data) {
  LocalDate localDate = toLocalDate(data.date);
  return data.dateTimeFormatter.format(localDate);
}

private LocalDate toLocalDate(Date date) {
  return date.toInstant().atZone(ZoneId.systemDefault()).toLocalDate();
}

Время (нс):


Память (байт):


Таким образом преобразование Date -> LocalDate выгодно при использовании "девятки". На "восьмёрке" затраты на преобразование сожрут всё преимущество DateTimeFormatter-a.


Бенчмарк


Вывод: используйте преимущества новых решений.


Ещё "восьмёрка"


В этом коде видим очевидную избыточность:


Iterator<Long> iterator = items // ArrayList<Integer>
  .stream()
  .map(Long::valueOf)
  .collect(toList())  // <---- зачем нам промежуточный список?
  .iterator();

while (iterator.hasNext()) {
  bh.consume(iterator.next());
}

Уберём её:


Iterator<Long> iterator = items // ArrayList<Integer>
  .stream()
  .map(Long::valueOf)
  .iterator();

while (iterator.hasNext()) {
  bh.consume(iterator.next());
}

Посмотрим, насколько улучшилась производительность:


Сравним с девяткой:


Удивительно, не так ли? Выше я утверждал, что лишняя работа ухудшает производительность. Но вот мы убираем лишнее — и (внезапно) становится хуже. Чтобы разобраться в происходящем, возьмём два итератора и посмотрим на них под лупой:


Раскрыть
Iterator iterator1 = items.stream().collect(toList()).iterator();
Iterator iterator2 = items.stream().iterator();


Первый итератор — это обычный ArrayList$Itr.


Проход по нему простой:
public boolean hasNext() {
  return cursor != size;
}

public E next() {
  checkForComodification();
  int i = cursor;
  if (i >= size) {
    throw new NoSuchElementException();
  }
  Object[] elementData = ArrayList.this.elementData;
  if (i >= elementData.length) {
    throw new ConcurrentModificationException();
  }
  cursor = i + 1;
  return (E) elementData[lastRet = i];
}


Второй любопытнее, это Spliterators$Adapter, в основе которого — ArrayList$ArrayListSpliterator.


Проход по нему сложнее
// java.util.Spliterators$Adapter

public boolean hasNext() {
  if (!valueReady)
    spliterator.tryAdvance(this);
  return valueReady;
}

public T next() {
  if (!valueReady && !hasNext())
    throw new NoSuchElementException();
  else {
    valueReady = false;
    return nextElement;
  }
}


Посмотрим на перебор итератором через async-profiler:


15.64%  j.u.ArrayList$ArrayListSpliterator.tryAdvance
10.67%  j.u.s.SpinedBuffer.clear
 9.86%  j.u.Spliterators$1Adapter.hasNext
 8.81%  j.u.s.StreamSpliterators$AbstractWrappingSpliterator.fillBuffer
 6.01%  o.o.j.i.Blackhole.consume
 5.71%  j.u.s.ReferencePipeline$3$1.accept
 5.57%  j.u.s.SpinedBuffer.accept
 5.06%  c.l.l.b.ir.IteratorFromStreamBenchmark.iteratorFromStream
 4.80%  j.l.Long.valueOf
 4.53%  c.l.l.b.i.IteratorFromStreamBenchmark$$Lambda$8.885721577.apply

Видно, что большая часть времени затрачивается на проход по итератору, хотя по большому счёту, он нам не нужен, ведь перебор можно сделать и так:


items
  .stream()
  .map(Long::valueOf)
  .forEach(bh::consume);

Сравним с остальными:


Stream::forEach явно в выигрыше, а ведь это странно: в основе по-прежнему лежит ArrayListSpliterator, но производительность его использования существенно улучшилась.


Посмотрим профиль:
29.04%  o.o.j.i.Blackhole.consume
22.92%  j.u.ArrayList$ArrayListSpliterator.forEachRemaining
14.47%  j.u.s.ReferencePipeline$3$1.accept
 8.79%  j.l.Long.valueOf
 5.37%  c.l.l.b.i.IteratorFromStreamBenchmark$$Lambda$9.617691115.accept
 4.84%  c.l.l.b.i.IteratorFromStreamBenchmark$$Lambda$8.1964917002.apply
 4.43%  j.u.s.ForEachOps$ForEachOp$OfRef.accept
 4.17%  j.u.s.Sink$ChainedReference.end
 1.27%  j.l.Integer.longValue
 0.53%  j.u.s.ReferencePipeline.map

В этом профиле большая часть времени затрачивается на "проглатывания" значения внутри Blackhole. По сравнению с итератором значительно большая часть времени затрачивается непосредственно на исполнение ява-кода. Можно предположить, что причиной является меньший удельный вес сборки мусора, по сравнению с перебором итератором. Проверим:


forEach:·gc.alloc.rate.norm                 100  avgt   30  216,001 ± 0,002  B/op
iteratorFromStream:·gc.alloc.rate.norm      100  avgt   30  416,004 ± 0,006  B/op

И действительно, Stream::forEach даёт вдвое меньшее потребление памяти.


Почему стало быстрее?

Цепочка вызовов от начала и до "чёрной дыры" выглядит так:



Как видим, из цепочки пропал вызов ArrayListSpliterator::tryAdvance, а вместо него появился ArrayListSpliterator::forEachRemaining:


// ArrayListSpliterator
public void forEachRemaining(Consumer<? super E> action) {
  int i, hi, mc; // hoist accesses and checks from loop
  ArrayList<E> lst; Object[] a;
  if (action == null)
    throw new NullPointerException();
  if ((lst = list) != null && (a = lst.elementData) != null) {
    if ((hi = fence) < 0) {
      mc = lst.modCount;
      hi = lst.size;
    }
    else
      mc = expectedModCount;
    if ((i = index) >= 0 && (index = hi) <= a.length) {
      for (; i < hi; ++i) {
        @SuppressWarnings("unchecked") E e = (E) a[i];             // <----
        action.accept(e);
      }
      if (lst.modCount == mc)
        return;
    }
  }
  throw new ConcurrentModificationException();
}

Высокая скорость ArrayListSpliterator::forEachRemaining достигается использованием прохода по всему массиву за 1 вызов метода. При использовании итератора проход ограничен одним элементом, поэтому мы всё время упираемся в ArrayListSpliterator::tryAdvance.
ArrayListSpliterator::forEachRemaining имеет доступ ко всему массиву и перебирает его счётным циклом без дополнительных вызовов.


Важное замечание

Обратите внимание, что механическая замена


Iterator<Long> iterator = items
  .stream()
  .map(Long::valueOf)
  .collect(toList())
  .iterator();

while (iterator.hasNext()) {
  bh.consume(iterator.next());
}

на


items
  .stream()
  .map(Long::valueOf)
  .forEach(bh::consume);

не всегда равнозначна, т. к. в первом случае мы используем для прохода копию данных, не затрагивая сам стрим, а во втором данные берутся непосредственно из стрима.


Бенчмарк


Вывод: имея дело со сложными представлениями данных будьте готовы к тому, что даже "железные" правила (лишняя работа вредит) перестают работать. Пример выше показывает, что кажущийся лишним промежуточный список даёт преимущество в виде более быстрой реализации перебора.


Два подвоха


StackTraceElement[] trace = th.getStackTrace();
StackTraceElement[] newTrace = Arrays
  .asList(trace)
  .subList(0, depth)
  .toArray(new StackTraceElement[newDepth]);  // <----

Первое, что бросается в глаза — протухшее "улучшение", а именно передача в метод Collection::toArray массива ненулевой длины. Здесь очень обстоятельно рассказывается почему это вредит.


Вторая проблема не столь очевидна, и для её понимания можно провести параллель между работой ревьюера и историка.


Вот это об этом пишет Робин Коллингвуд:

Предположим, например, что он [историк] читает Кодекс Феодосия и перед ним – эдикт императора. Простое чтение слов и возможность их перевести еще не равносильны пониманию их исторического значения. Чтобы оценить его, историк должен представить себе ситуацию, которую пытался разрешить император...


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


1) список заворачивается в массив
2) часть списка отрезается
3) отрезанный подсписок преобразуется обратно в массив


Иными словами разработчик намеревался создать подмассив, а это гораздо проще сделать так:


StackTraceElement[] trace = th.getStackTrace();
StackTraceElement[] newTrace = Arrays.copyOf(trace, depth);     // неявный отсчёт с 0
// или
StackTraceElement[] newTrace = Arrays.copyOfRange(trace, 0, depth); //явный отсчёт с 0

Сравним (нс):


Бенчмарк


Стримоз головного мозга


List<T> list = getList();
Set<T> set = getSet();
return list.stream().allMatch(set::contains); // что проверяется на самом деле?

Проверяется вхождение всех элементов списка в набор, но этот код можно значительно упростить, одновременно улучшив его производительность:


List<T> list = getList();
Set<T> set = getSet();
return set.containsAll(list);

Сравним (нс):


Бенчмарк


Ненужные обёртки


Есть интерфейс:


interface FileNameLoader {
  String[] loadFileNames();
}

и использование некой его реализации:


private FileNameLoader loader;

void load() {
  for (String str : asList(loader.loadFileNames())) {  // <---- ненужное заворачивание
    use(str);
  }
}

Часто разработчики считают, что сахарок forEach используется только с коллекциями, хотя с массивом его тоже можно:


private FileNameLoader loader;

void load() {
  for (String str : loader.loadFileNames()) {  // <---- проще и быстрее 
    use(str);
  }
}

Сравним (нс):


Бенчмарк


Полезное для разработчика

https://youtrack.jetbrains.com/issue/IDEA-182206
IDEA-182206 Simplification for Arrays.asList().subList().toArray()
Исправлено в 2018.1


https://youtrack.jetbrains.com/issue/IDEA-180847
IDEA-180847 Inspection 'Call to Collection.toArray with zero-length array argument' brings pessimization
Исправлено в 2018.1


https://youtrack.jetbrains.com/issue/IDEA-181928
IDEA-181928 Stream.allMatch(Collection::contains) can be simplified to Collection.containsAll()
Исправлено в 2018.1


https://youtrack.jetbrains.com/issue/IDEA-184240
IDEA-184240 Unnecessary array-to-collection wrapping should be detected
Исправлено в 2018.1


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


  • короче
  • надёжнее
  • проще и понятнее

Если вы на своём проекте выловили антипаттерн, который можно формализовать, не стесняйтесь сообщать о них разработчикам систем статического анализа, а не только вашим товарищам. Таким образом, вы поможете всем: разработчики добавят новую проверку и пропылесосят код самой "Идеи" (и вы получите более отказоустойчивую и быструю среду разработки), разработчики библиотек и фреймворков возьмут новую "Идею" и почистят свой код (и вы получите улучшенные библиотеки и фреймворки), и наконец вы сами можете применить статический анализ.


> Исходники замеров с пояснениями


> Папка с замерами

Комментарии (12)


  1. mayorovp
    10.10.2018 12:43

    Метод java.lang.String::equals интринзифицирован, поэтому познакового сравнения при исполнении не происходит.

    А в чем неточность-то? Или JIT сравнивает строки не точно так же познаково, пусть и более быстро?


    1. tsypanov Автор
      10.10.2018 13:05

      Я думал, что исполняться будет ява-кода из `String::equals`, но интринзифицированный метод, насколько я понимаю, исполняет нативный код, оптимизированный под конкретную платформу.


      1. mayorovp
        10.10.2018 13:11

        Тем не менее, оптимизация тут всего лишь означает уменьшение константы. Магии не бывает, и сравнить две строки без побайтового либо посимвольного прохода по ним невозможно.


        1. lany
          10.10.2018 13:21

          Там векторизуется неплохо для длинных строк, но всё равно, конечно О(N). Начиная с Java 9, кстати, есть интересный частный случай: если в одной строке все символы Latin1, а в другой — нет, то сразу скажет, что не равны.


        1. Perlovich
          10.10.2018 13:25

          сравнить две строки без побайтового либо посимвольного прохода по ним невозможно.


          Можно. Посмотрите на String::intern.

          Вполне можно добиться того, что в куче не будет одинаковых строк. Т.е., если в разных местах в коде использовать одинаковое строковое значение, то оно будет указывать на один и тот же объект.

          Я не знаю, насколько это поведение надежно, но данные оптимизации действительно имеют место быть при компиляции. Хотя я лично на них не полагаюсь, и пишу в коде string1.equals(string2) вместо string1 == string2. Но второй вариант тоже, в теории, будет работать достаточно стабильно.


          1. mayorovp
            10.10.2018 13:29

            Метод String::intern не вызывается автоматически для всех конструируемых строк, а оптимизации при компиляции применяются только к строковым литералам.


            Так что второй вариант является ошибкой, и не должен работать ни в теории, ни на практике. Вот пример: https://ideone.com/c0OYGW


          1. poxvuibr
            11.10.2018 10:18
            +2

            Можно. Посмотрите на String::intern.

            Не надо пользоваться String::intern! С восклицательным знаком не надо. Хотите снизить потребление памяти строками в какой-то операции — складывайте строки в Map, и оттуда же забирайте.


  1. Anton23
    10.10.2018 12:54

    Жалко, что больше не статей, кроме вашей. По крайне мере я не нашел, только презентации, или YT видео.


    1. tsypanov Автор
      10.10.2018 13:13

      Есть крутые доклады Шипилёва и Куксенко, они об общих началах улучшений производительности и о низкоуровневых вещах. Универсальную статью написать сложно, т. к. примеры из этой статьи, основаны на нашем конкретном проекте и совсем не обязательно подойдут для вашего. Как мне кажется, именно поэтому тот же Шипилёв, Черёмин или Вакарт в основном рассказывают об улучшениях кода, привязанного только к JDK как к некой постоянной для многих разработчиков.


      1. Anton23
        10.10.2018 13:19

        Я не про это говорил. Не о применимости докладов, а о том, что если были бы текстовые версии докладов, было бы лучше.


        1. lany
          10.10.2018 13:22

          JUGRu десятками публикует расшифровки докладов с конференций
          habr.com/company/jugru/blog Только там рыться надо, потому что они тонну всего помимо этого публикуют. Вот, например, из недавнего habr.com/company/jugru/blog/424763


          1. Anton23
            10.10.2018 13:41

            Упс. Вы правы. Но все таки, некоторых статей нет.