hashCode() - how to implement it? Where is it used?
Method hashCode() is inherited from Object class. Its purpose is to return an int value, which can be used to represent given object. There is a contract between hashCode() and equals() which says that whenever equals() method used to compare two objects returns true, then values returned by hashCode() functions called on those objects must be the same. Remember that it does not work like that vice versa - when hashCode() values are the same, then it does not mean that equals() invoked for those two objects will always return true. hashCode() method is widely used by java containers like HashMap and HashSet. That is why it is desired that hashCode() will return unique values for unique objects as often as it is possible since it can improve performace of these containers (however it is not concidered as an error when hashCode() returns the same value for different objects).
The basic implementation of hashCode() uses object address as returned value. The conclusion is - whenever you create your own class, and you want hashCode() to return unique values for unique objects as often as it is possible, then you will have to implement it. You will see that in the example below.
Lets create a Car class:
Now lets compare hashCode() values. The cars have the same content (fields) so we would expect hashCode() values of both car objects to be the same, since these objects are not unique.
Unfortunatelly, hashCode() methods are returning different values. Basic implementation inherited from Object class leads to comparison of object addresses, which are not the same, so we receive false value. It breaks the contract with equals() method which returns true in this case.
To fix this issue we have to implement hashCode() method in our Car class. To do that properly we have to take into consideration all of the fields in the class. IntelliJ, Eclipse and NetBeans have their methods to auto-generate hashCode() methods. Below is the example generated with IntelliJ :
Now hashCode() invoked for the same Car instances will return the same values.
How to implement it?
Whenever you override an equals()method, you will also have to override hashCode() method to fulfill the contract between them (when equals() used to compare two objects returns true, then values returned by hashCode() for those objects must be the same).The basic implementation of hashCode() uses object address as returned value. The conclusion is - whenever you create your own class, and you want hashCode() to return unique values for unique objects as often as it is possible, then you will have to implement it. You will see that in the example below.
Lets create a Car class:
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
class Car { | |
private String type; | |
private BigDecimal price; | |
public Car(String type, BigDecimal price) { | |
this.type = type; | |
this.price = price; | |
} | |
public String getType() { | |
return type; | |
} | |
public void setType(String type) { | |
this.type = type; | |
} | |
public BigDecimal getPrice() { | |
return price; | |
} | |
public void setPrice(BigDecimal price) { | |
this.price = price; | |
} | |
@Override | |
public boolean equals(Object o) { | |
if (o == this) return true; | |
if (!(o instanceof Car)) { | |
return false; | |
} | |
Car car = (Car) o; | |
return car.type.equals(type) && | |
car.price.equals(price); | |
} | |
} |
Now lets compare hashCode() values. The cars have the same content (fields) so we would expect hashCode() values of both car objects to be the same, since these objects are not unique.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
public static void main(String[] args) { | |
Car fiat1 = new Car("Fiat", new BigDecimal(4000)); | |
Car fiat2 = new Car("Fiat", new BigDecimal(4000)); | |
System.out.println(fiat2.hashCode() == fiat1.hashCode()); //false | |
System.out.println(fiat2.equals(fiat1)); //true | |
} |
Unfortunatelly, hashCode() methods are returning different values. Basic implementation inherited from Object class leads to comparison of object addresses, which are not the same, so we receive false value. It breaks the contract with equals() method which returns true in this case.
To fix this issue we have to implement hashCode() method in our Car class. To do that properly we have to take into consideration all of the fields in the class. IntelliJ, Eclipse and NetBeans have their methods to auto-generate hashCode() methods. Below is the example generated with IntelliJ :
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
@Override | |
public int hashCode() { | |
return Objects.hash(getType(), getPrice()); | |
} |
Komentarze
Prześlij komentarz