- Java equals hashcode связь
- Java Challengers #4: Сравнение объектов с equals() и hashCode()
- Переопределение equals() и hashCode()
- Сравнение объектов с equals()
- Анализ вариантов сравнения объектов
- equals() в сравнении с ==
- Идентификация объектов с hashCode()
- Использование equals() и hashCode() с коллекциями
- Рекомендации по использованию equals() и hashCode()
- Правила сравнения объектов
- Решите задачку на equals() и hashCode()
- Что произошло? Понимание equals() и hashCode()
- Ответ
- Частые ошибки с equals() и hashCode()
- Что нужно помнить о equals() и hashCode()
- Изучите больше о Java
Java equals hashcode связь
День добрый, уважаемые дамы и господа знатоки. Есть вопрос прикладного характера. В начале объясняется, что прежде сравнения нужно убедиться, что сравниваются объекты одного класса. Это логично и понятно. Соответствующая строка в коде дополняет сказанное. НО! Зачем тогда следующей строкой приводить проверяемый объект к классу объекта-эталона? Если мы и так знаем, что объекты одного класса, иначе до выполнения этой строки не дошло бы — метод прекращается после return, зачем ещё раз приводить их к одному классу? Выглядит дико и просто как пятое колесо. У велосипеда. @Override public boolean equals(Object o) < if (getClass() != o.getClass()) return false; Man man = (Man) o; // Вот это что?? Зачем. return dnaCode == man.dnaCode; >
@Override public boolean equals(Object o) < if (this == o) return true; if (o == null || getClass() != o.getClass()) return false; LuxuryAuto that = (LuxuryAuto) o; // а может добавить еще одну проверку ? if (this.hashCode() != o.hashCode()) return false; // например такую ? // тогда псевдо одинаковые обьекты будут не равны при не равных хешах if (manufactureYear != that.manufactureYear) return false; return dollarPrice == that.dollarPrice; >@Override public int hashCode()
последний пример вводит в тупик. При сравнении через equals, сначала проверяются hashCode’ы, если они не равны, то сравнение через equals не последует. В нашем примере как раз таки это и происходит. Как equals показал true, если хэши у них разные и до метода equals код не должен был дойти и сразу дать false?? Теперь поговорим о методе hashCode(). Зачем он нужен? Ровно для той же цели — сравнения объектов. Но ведь у нас уже есть equals()! Зачем же еще один метод? Ответ прост: для повышения производительности. Хэш-функция, которая представлена в Java методом hashCode(), возвращает числовое значение фиксированной длины для любого объекта. В случае с Java метод hashCode() возвращает для любого объекта 32-битное число типа int. Сравнить два числа между собой — гораздо быстрее, чем сравнить два объекта методом equals(), особенно если в нем используется много полей. Если в нашей программе будут сравниваться объекты, гораздо проще сделать это по хэш-коду, и только если они равны по hashCode() — переходить к сравнению по equals().
Java Challengers #4: Сравнение объектов с equals() и hashCode()
В этой статье вы узнаете, как связаны между собой методы equals() и hashCode() и как они используются при сравнении объектов.
Без использования equals() и hashCode() для сравнения состояния двух объектов нам нужно писать много сравнений » if «, сравнивая каждое поле объекта. Такой подход делает код запутанным и трудным для чтения. Работая вместе, эти два метода помогают создавать более гибкий и согласованный код.
Исходный код для статьи находится здесь.
Переопределение equals() и hashCode()
Переопределение метода (method overriding) — это приём при котором поведение родительского класса или интерфейса переписывается (переопределяется) в подклассе (см. Java Challengers #3: Полиморфизм и наследование, анг.). В Java у каждого объекта есть методы equals() и hashCode() и для правильной работы они должны быть переопределены.
Чтобы понять, как работает переопределение equals() и hashCode() , изучим их реализацию в базовых классах Java. Ниже приведён метод equals() класса Object . Метод проверяет, совпадает ли текущий экземпляр с переданным объектом obj .
public boolean equals(Object obj)
Теперь посмотрим на метод hashCode() в классе Object .
@HotSpotIntrinsicCandidate public native int hashCode();
Это native — метод, который написан на другом языке, таком как Си, и он возвращает некоторый числовой код, связанный с адресом памяти объекта. (Если вы не пишете код JDK, то не важно точно знать, как работает этот метод.)
Примечание переводчика: про значение, связанное с адресом сказано не совсем корректно (спасибо vladimir_dolzhenko). В HotSpot JVM по умолчанию используются псевдослучайные числа. Описание реализации hashCode() для HotSpot, есть здесь и здесь.
Если методы equals() и hashCode() не переопределены, вместо них будут вызваны методы класса Object , описанные выше. В этом случае методы не выполняют реальной цели equals() и hashCode() , которая состоит в том, чтобы проверить, имеют ли объекты одинаковые состояния.
Как правило, при переопределении equals() также переопределяется hashCode() .
Сравнение объектов с equals()
Метод equals() используется для сравнения объектов. Чтобы определить одинаковые объекты или нет, equals() сравнивает значения полей объектов:
public class EqualsAndHashCodeExample < public static void main(String. args)< System.out.println(new Simpson("Homer", 35, 120) .equals(new Simpson("Homer",35,120))); System.out.println(new Simpson("Bart", 10, 120) .equals(new Simpson("El Barto", 10, 45))); System.out.println(new Simpson("Lisa", 54, 60) .equals(new Object())); >static class Simpson < private String name; private int age; private int weight; public Simpson(String name, int age, int weight) < this.name = name; this.age = age; this.weight = weight; >@Override public boolean equals(Object o) < // 1 if (this == o) < return true; >// 2 if (o == null || getClass() != o.getClass()) < return false; >// 3 Simpson simpson = (Simpson) o; return age == simpson.age && weight == simpson.weight && name.equals(simpson.name); > > >
Посмотрим на метод equals() . Первое сравнение сравнивает текущий экземпляр объекта this с переданным объектом o . Если это один и тот же объект, то equals() вернёт true .
Во втором сравнении проверяется, является ли переданный объект null и какой у него тип. Если переданный объект другого типа, то объекты не равны.
Наконец, equals() сравнивает поля объектов. Если два объекта имеют одинаковые значения полей, то объекты совпадают.
Анализ вариантов сравнения объектов
Теперь давайте посмотрим на варианты сравнений объектов в методе main() . Сначала мы сравниваем два объекта Simpson :
System.out.println( new Simpson("Homer", 35, 120).equals( new Simpson("Homer", 35, 120)));
У полей этих объектов одинаковые значения, поэтому результат будет true .
Затем снова сравниваем два объекта Simpson :
System.out.println( new Simpson("Bart", 10, 45).equals( new Simpson("El Barto", 10, 45)));
Объекты здесь похожи, но значения имён разные: Bart и El Barto. Поэтому результат будет false .
Наконец, давайте сравним объект Simpson и экземпляр класса Object :
System.out.println( new Simpson("Lisa", 54, 60).equals( new Object()));
В этом случае результат будет false , так как типы объектов отличаются.
equals() в сравнении с ==
На первый взгляд кажется, что оператор == и метод equals() делают одно и то же, но, на самом деле, они работают по-разному. Оператор == сравнивает, указывают ли две ссылки на один и тот же объект. Например:
Simpson homer = new Simpson("Homer", 35, 120); Simpson homer2 = new Simpson("Homer", 35, 120); System.out.println(homer == homer2);
Мы создали два разных экземпляра Simpson с помощью оператора new . Поэтому переменные homer и homer2 будут указывать на разные объекты в куче. Таким образом, в результате получим false .
Во следующем примере используем переопределенный метод equals() :
System.out.println(homer.equals(homer2));
В этом случае будут сравниваться поля. Поскольку значения полей у обоих объектов Simpson одинаковые, результат будет true .
Идентификация объектов с hashCode()
Для оптимизации производительности при сравнении объектов используется метод hashCode() . Метод hashCode() возвращает уникальный идентификатор для каждого объекта, что упрощает сравнение состояний объектов.
Если хэш-код объекта не совпадает с хэш-кодом другого объекта, то можно не выполнять метод equals() : вы просто знаете, что два объекта не совпадают. С другой стороны, если хэш-код одинаковый то, необходимо выполнить метод equals() , чтобы определить, совпадают ли значения полей.
Рассмотрим практический пример с hashCode() .
public class HashcodeConcept < public static void main(String. args) < Simpson homer = new Simpson(1, "Homer"); Simpson bart = new Simpson(2, "Homer"); boolean isHashcodeEquals = homer.hashCode() == bart.hashCode(); if (isHashcodeEquals) < System.out.println("Следует сравнить методом equals."); >else < System.out.println("Не следует сравнивать методом equals, т.к. " + "идентификатор отличается, что означает, что объекты точно не равны."); >> static class Simpson < int id; String name; public Simpson(int id, String name) < this.id = id; this.name = name; >@Override public boolean equals(Object o) < if (this == o) return true; if (o == null || getClass() != o.getClass()) return false; Simpson simpson = (Simpson) o; return simpson.id && name.equals(simpson.name); >@Override public int hashCode() < return id; >> >
Метод hashCode() , который всегда возвращает одно и то же значение, допустим, но не эффективен. В этом случае сравнение всегда будет возвращать true , поэтому метод equals() будет выполняться всегда. В этом случае нет никакого улучшения производительности.
Использование equals() и hashCode() с коллекциями
Классы, реализующие интерфейс Set (множество) должны не допускать добавления повторяющихся элементов. Ниже приведены некоторые классы, реализующие интерфейс Set :
В Set могут быть добавлены только уникальные элементы. Таким образом, если вы хотите добавить элемент, например, в HashSet , вы должны использовать сначала методы equals() и hashCode() , чтобы убедиться, что этот элемент уникальный. Если методы equals() и hashCode() не были переопределены, вы рискуете вставить повторяющиеся значения.
Посмотрим на часть реализации метода add() в HashSet :
Перед добавлением нового элемента HashSet проверяет, существует ли элемент в данной коллекции. Если объект совпадает, то новый элемент вставляться не будет.
Методы equals() и hashCode() используются не только в Set . Также эти методы требуются для HashMap, Hashtable, и LinkedHashMap. Как правило, если вы видите коллекцию с префиксом «Hash», вы можете быть уверены, что для её корректной работы требуется переопределение методов hashCode() и equals() .
Рекомендации по использованию equals() и hashCode()
Выполняйте метод equals() только для объектов с одинаковым хэш-кодом. Не выполняйте equals() , если хэш-код отличается.
Таблица 1. Сравнение хэш-кодов
Если сравнение hashCode() . | То . |
---|---|
возвращает true | выполнить equals() |
возвращает false | не выполнять equals() |
Этот принцип в основном используется в коллекциях Set или Hash по соображениям производительности.
Правила сравнения объектов
Когда сравнение hashCode() возвращает false , метод equals() также должен возвращать false . Если хэш-код отличается, то объекты определенно не равны.
Таблица 2.Сравнение объектов с hashCode()
Когда сравнение hashCode() возвращает . | Метод equals() должен вернуть . |
---|---|
true | true или false |
false | false |
Когда метод equals() возвращает true , это означает, что объекты равны во всех значениях и атрибутах. В этом случае сравнение хэш-кода также должно быть истинным.
Таблица 3. Сравнение объектов с equals()
Когда метод equals() возвращает . | Метод hashCode() должен вернуть . |
---|---|
true | true |
false | true или false |
Решите задачку на equals() и hashCode()
Пришло время проверить ваши знания методов equals() и hashCode() . Задача состоит в том, чтобы выяснить результат нескольких equals() и итоговый размер коллекции Set .
Для начала, внимательно изучите следующий код :
public class EqualsHashCodeChallenge < public static void main(String. args) < System.out.println(new Simpson("Bart").equals(new Simpson("Bart"))); Simpson overriddenHomer = new Simpson("Homer") < public int hashCode() < return (43 + 777) + 1; >>; System.out.println(new Simpson("Homer").equals(overriddenHomer)); Set set = new HashSet(Set.of(new Simpson("Homer"), new Simpson("Marge"))); set.add(new Simpson("Homer")); set.add(overriddenHomer); System.out.println(set.size()); > static class Simpson < String name; Simpson(String name) < this.name = name; >@Override public boolean equals(Object obj) < Simpson otherSimpson = (Simpson) obj; return this.name.equals(otherSimpson.name) && this.hashCode() == otherSimpson.hashCode(); >@Override public int hashCode() < return (43 + 777); >> >
Сначала проанализируйте код, подумайте, какой будет результат. И только потом запустите код. Цель в том, чтобы улучшить ваши навыки анализа кода и усвоить основные концепции Java, чтобы вы могли сделать свой код лучше.
A) true true 4 B) true false 3 C) true false 2 D) false true 3
Что произошло? Понимание equals() и hashCode()
В первом сравнении результат equals() равен true , поскольку состояния объектов одинаковые, и метод hashCode() возвращает одно и то же значнеие для обоих объектов.
Во втором сравнении для переменной overriddenHomer был переопределён метод hashCode() . Для обоих объектов Simpson имя равно «Homer», но для overriddenHomer метод hashCode() возвращает другое значение. В этом случае результат метода equals() будет false , так как в нём содержится сравнение с хэш-кодом.
Вы, должно быть, поняли, что в коллекции будет три объекта Simpson . Давайте разберём это.
Первый объект в наборе будет вставлен как обычно:
new Simpson("Homer"); // Добавляется
Следующий объект также будет вставлен в обычном порядке, поскольку содержит значение, отличное от предыдущего объекта:
new Simpson("Marge"); // Добавляется
Наконец, следующий объект Simpson имеет то же значение имени, что и первый объект. В этом случае объект вставляться не будет:
set.add(new Simpson("Homer")); // Не добавляется
Как мы знаем, объект overridenHomer использует другое значение хэш-кода в отличие от обычного экземпляра Simpson(«Homer») . По этой причине этот элемент будет вставлен в коллекцию:
set.add(overriddenHomer); // Добавляется
Ответ
Правильный ответ — B. Вывод будет:
Частые ошибки с equals() и hashCode()
- Отсутствие переопределения hashCode() вместе с переопределением equals() или наоборот.
- Отсутствие переопределения equals() и hashCode() при использовании хэш-коллекций, таких как HashSet .
- Возврат постоянного значения в методе hashCode() вместо возврата уникального кода для каждого объекта.
- Равнозначное использование == и equals() . Оператор == сравнивает ссылки на объекты, тогда как метод equals() сравнивает значения объектов.
Что нужно помнить о equals() и hashCode()
- Рекомендуется всегда переопределять методы equals() and hashCode() в ваших POJO (рус., анг.)
- Используйте эффективный алгоритм для создания уникального хэш-кода.
- При переопределении метода equals() всегда переопределяйте метод hashCode() .
- Метод equals() должен сравнивать полное состояние объектов (значения из полей).
- Метод hashCode() может быть идентификатором (ID) POJO.
- Если результат сравнения хэш-кода двух объектов false , то метод equals() также должен иметь значение false .
- Если equals() и hashCode() не переопределяются при использовании хэш-коллекций, то коллекция будет иметь повторяющиеся элементы.
Изучите больше о Java
Традиционно жду ваши комментарии и приглашаю на открытый урок, который уже 18 марта проведет наш преподаватель Сергей Петрелевич