Class EqualsBuilder
Object.equals(Object)
methods.
This class provides methods to build a good equals method for any
class. It follows rules laid out in
Effective Java
, by Joshua Bloch. In particular the rule for comparing doubles
,
floats
, and arrays can be tricky. Also, making sure that
equals()
and hashCode()
are consistent can be
difficult.
Two Objects that compare as equals must generate the same hash code, but two Objects with the same hash code do not have to be equal.
All relevant fields should be included in the calculation of equals. Derived fields may be ignored. In particular, any field used in generating a hash code must be used in the equals method, and vice versa.
Typical use for the code is as follows:
public boolean equals(Object obj) { if (obj == null) { return false; } if (obj == this) { return true; } if (obj.getClass() != getClass()) { return false; } MyClass rhs = (MyClass) obj; return new EqualsBuilder() .appendSuper(super.equals(obj)) .append(field1, rhs.field1) .append(field2, rhs.field2) .append(field3, rhs.field3) .isEquals(); }
Alternatively, there is a method that uses reflection to determine
the fields to test. Because these fields are usually private, the method,
reflectionEquals
, uses AccessibleObject.setAccessible
to
change the visibility of the fields. This will fail under a security
manager, unless the appropriate permissions are set up correctly. It is
also slower than testing explicitly. Non-primitive fields are compared using
equals()
.
A typical invocation for this method would look like:
public boolean equals(Object obj) { return EqualsBuilder.reflectionEquals(this, obj); }
The EqualsExclude
annotation can be used to exclude fields from being
used by the reflectionEquals
methods.
- Since:
- 1.0
-
Constructor Summary
-
Method Summary
Modifier and TypeMethodDescriptionappend
(boolean[] lhs, boolean[] rhs) Deep comparison of array ofboolean
.append
(boolean lhs, boolean rhs) Test if twobooleans
s are equal.append
(byte[] lhs, byte[] rhs) Deep comparison of array ofbyte
.append
(byte lhs, byte rhs) Test if twobyte
s are equal.append
(char[] lhs, char[] rhs) Deep comparison of array ofchar
.append
(char lhs, char rhs) Test if twochar
s are equal.append
(double[] lhs, double[] rhs) Deep comparison of array ofdouble
.append
(double lhs, double rhs) Test if twodouble
s are equal by testing that the pattern of bits returned bydoubleToLong
are equal.append
(float[] lhs, float[] rhs) Deep comparison of array offloat
.append
(float lhs, float rhs) Test if twofloat
s are equal by testing that the pattern of bits returned by doubleToLong are equal.append
(int[] lhs, int[] rhs) Deep comparison of array ofint
.append
(int lhs, int rhs) Test if twoint
s are equal.append
(long[] lhs, long[] rhs) Deep comparison of array oflong
.append
(long lhs, long rhs) Test if twolong
s are equal.append
(short[] lhs, short[] rhs) Deep comparison of array ofshort
.append
(short lhs, short rhs) Test if twoshort
s are equal.Performs a deep comparison of twoObject
arrays.Test if twoObject
s are equal using either #reflectionAppend(Object, Object)
, if object are non primitives (or wrapper of primitives) or if fieldtestRecursive
is set tofalse
.appendSuper
(boolean superEquals) Adds the result ofsuper.equals()
to this builder.build()
Returnstrue
if the fields that have been checked are all equal.boolean
isEquals()
Returnstrue
if the fields that have been checked are all equal.reflectionAppend
(Object lhs, Object rhs) Tests if twoobjects
by using reflection.static boolean
reflectionEquals
(Object lhs, Object rhs, boolean testTransients) This method uses reflection to determine if the twoObject
s are equal.static boolean
reflectionEquals
(Object lhs, Object rhs, boolean testTransients, Class<?> reflectUpToClass, boolean testRecursive, String... excludeFields) This method uses reflection to determine if the twoObject
s are equal.static boolean
reflectionEquals
(Object lhs, Object rhs, boolean testTransients, Class<?> reflectUpToClass, String... excludeFields) This method uses reflection to determine if the twoObject
s are equal.static boolean
reflectionEquals
(Object lhs, Object rhs, String... excludeFields) This method uses reflection to determine if the twoObject
s are equal.static boolean
reflectionEquals
(Object lhs, Object rhs, Collection<String> excludeFields) This method uses reflection to determine if the twoObject
s are equal.void
reset()
Reset the EqualsBuilder so you can use the same object again.setBypassReflectionClasses
(List<Class<?>> bypassReflectionClasses) SetsClass
es whose instances should be compared by calling theirequals
although being in recursive mode.protected void
setEquals
(boolean isEquals) Sets theisEquals
value.setExcludeFields
(String... excludeFields) Sets field names to be excluded by reflection tests.setReflectUpToClass
(Class<?> reflectUpToClass) Sets the superclass to reflect up to at reflective tests.setTestRecursive
(boolean testRecursive) Sets whether to test fields recursively, instead of using their equals method, when reflectively comparing objects.setTestTransients
(boolean testTransients) Sets whether to include transient fields when reflectively comparing objects.
-
Constructor Details
-
EqualsBuilder
public EqualsBuilder()Constructor for EqualsBuilder.Starts off assuming that equals is
true
.- See Also:
-
-
Method Details
-
reflectionEquals
This method uses reflection to determine if the twoObject
s are equal.It uses
AccessibleObject.setAccessible
to gain access to private fields. This means that it will throw a security exception if run under a security manager, if the permissions are not set up correctly. It is also not as efficient as testing explicitly. Non-primitive fields are compared usingequals()
.If the TestTransients parameter is set to
true
, transient members will be tested, otherwise they are ignored, as they are likely derived fields, and not part of the value of theObject
.Static fields will not be tested. Superclass fields will be included.
- Parameters:
lhs
-this
objectrhs
- the other objecttestTransients
- whether to include transient fields- Returns:
true
if the two Objects have tested equals.- See Also:
-
reflectionEquals
public static boolean reflectionEquals(Object lhs, Object rhs, boolean testTransients, Class<?> reflectUpToClass, boolean testRecursive, String... excludeFields) This method uses reflection to determine if the twoObject
s are equal.It uses
AccessibleObject.setAccessible
to gain access to private fields. This means that it will throw a security exception if run under a security manager, if the permissions are not set up correctly. It is also not as efficient as testing explicitly. Non-primitive fields are compared usingequals()
.If the testTransients parameter is set to
true
, transient members will be tested, otherwise they are ignored, as they are likely derived fields, and not part of the value of theObject
.Static fields will not be included. Superclass fields will be appended up to and including the specified superclass. A null superclass is treated as java.lang.Object.
If the testRecursive parameter is set to
true
, non primitive (and non primitive wrapper) field types will be compared byEqualsBuilder
recursively instead of invoking theirequals()
method. Leading to a deep reflection equals test.- Parameters:
lhs
-this
objectrhs
- the other objecttestTransients
- whether to include transient fieldsreflectUpToClass
- the superclass to reflect up to (inclusive), may benull
testRecursive
- whether to call reflection equals on non-primitive fields recursively.excludeFields
- array of field names to exclude from testing- Returns:
true
if the two Objects have tested equals.- Since:
- 3.6
- See Also:
-
reflectionEquals
public static boolean reflectionEquals(Object lhs, Object rhs, boolean testTransients, Class<?> reflectUpToClass, String... excludeFields) This method uses reflection to determine if the twoObject
s are equal.It uses
AccessibleObject.setAccessible
to gain access to private fields. This means that it will throw a security exception if run under a security manager, if the permissions are not set up correctly. It is also not as efficient as testing explicitly. Non-primitive fields are compared usingequals()
.If the testTransients parameter is set to
true
, transient members will be tested, otherwise they are ignored, as they are likely derived fields, and not part of the value of theObject
.Static fields will not be included. Superclass fields will be appended up to and including the specified superclass. A null superclass is treated as java.lang.Object.
- Parameters:
lhs
-this
objectrhs
- the other objecttestTransients
- whether to include transient fieldsreflectUpToClass
- the superclass to reflect up to (inclusive), may benull
excludeFields
- array of field names to exclude from testing- Returns:
true
if the two Objects have tested equals.- Since:
- 2.0
- See Also:
-
reflectionEquals
This method uses reflection to determine if the twoObject
s are equal.It uses
AccessibleObject.setAccessible
to gain access to private fields. This means that it will throw a security exception if run under a security manager, if the permissions are not set up correctly. It is also not as efficient as testing explicitly. Non-primitive fields are compared usingequals()
.Transient members will be not be tested, as they are likely derived fields, and not part of the value of the Object.
Static fields will not be tested. Superclass fields will be included.
- Parameters:
lhs
-this
objectrhs
- the other objectexcludeFields
- Collection of String field names to exclude from testing- Returns:
true
if the two Objects have tested equals.- See Also:
-
reflectionEquals
This method uses reflection to determine if the twoObject
s are equal.It uses
AccessibleObject.setAccessible
to gain access to private fields. This means that it will throw a security exception if run under a security manager, if the permissions are not set up correctly. It is also not as efficient as testing explicitly. Non-primitive fields are compared usingequals()
.Transient members will be not be tested, as they are likely derived fields, and not part of the value of the Object.
Static fields will not be tested. Superclass fields will be included.
- Parameters:
lhs
-this
objectrhs
- the other objectexcludeFields
- array of field names to exclude from testing- Returns:
true
if the two Objects have tested equals.- See Also:
-
append
Test if twobooleans
s are equal.- Parameters:
lhs
- the left-hand sideboolean
rhs
- the right-hand sideboolean
- Returns:
this
instance.
-
append
Deep comparison of array ofboolean
. Length and all values are compared.The method
append(boolean, boolean)
is used.- Parameters:
lhs
- the left-hand sideboolean[]
rhs
- the right-hand sideboolean[]
- Returns:
this
instance.
-
append
Test if twobyte
s are equal.- Parameters:
lhs
- the left-hand sidebyte
rhs
- the right-hand sidebyte
- Returns:
this
instance.
-
append
Deep comparison of array ofbyte
. Length and all values are compared.The method
append(byte, byte)
is used.- Parameters:
lhs
- the left-hand sidebyte[]
rhs
- the right-hand sidebyte[]
- Returns:
this
instance.
-
append
Test if twochar
s are equal.- Parameters:
lhs
- the left-hand sidechar
rhs
- the right-hand sidechar
- Returns:
this
instance.
-
append
Deep comparison of array ofchar
. Length and all values are compared.The method
append(char, char)
is used.- Parameters:
lhs
- the left-hand sidechar[]
rhs
- the right-hand sidechar[]
- Returns:
this
instance.
-
append
Test if twodouble
s are equal by testing that the pattern of bits returned bydoubleToLong
are equal.This handles NaNs, Infinities, and
-0.0
.It is compatible with the hash code generated by
HashCodeBuilder
.- Parameters:
lhs
- the left-hand sidedouble
rhs
- the right-hand sidedouble
- Returns:
this
instance.
-
append
Deep comparison of array ofdouble
. Length and all values are compared.The method
append(double, double)
is used.- Parameters:
lhs
- the left-hand sidedouble[]
rhs
- the right-hand sidedouble[]
- Returns:
this
instance.
-
append
Test if twofloat
s are equal by testing that the pattern of bits returned by doubleToLong are equal.This handles NaNs, Infinities, and
-0.0
.It is compatible with the hash code generated by
HashCodeBuilder
.- Parameters:
lhs
- the left-hand sidefloat
rhs
- the right-hand sidefloat
- Returns:
this
instance.
-
append
Deep comparison of array offloat
. Length and all values are compared.The method
append(float, float)
is used.- Parameters:
lhs
- the left-hand sidefloat[]
rhs
- the right-hand sidefloat[]
- Returns:
this
instance.
-
append
Test if twoint
s are equal.- Parameters:
lhs
- the left-hand sideint
rhs
- the right-hand sideint
- Returns:
this
instance.
-
append
Deep comparison of array ofint
. Length and all values are compared.The method
append(int, int)
is used.- Parameters:
lhs
- the left-hand sideint[]
rhs
- the right-hand sideint[]
- Returns:
this
instance.
-
append
Test if twolong
s are equal.- Parameters:
lhs
- the left-hand sidelong
rhs
- the right-hand sidelong
- Returns:
this
instance.
-
append
Deep comparison of array oflong
. Length and all values are compared.The method
append(long, long)
is used.- Parameters:
lhs
- the left-hand sidelong[]
rhs
- the right-hand sidelong[]
- Returns:
this
instance.
-
append
Test if twoObject
s are equal using either #reflectionAppend(Object, Object)
, if object are non primitives (or wrapper of primitives) or if fieldtestRecursive
is set tofalse
. Otherwise, using theirequals
method.- Parameters:
lhs
- the left-hand side objectrhs
- the right-hand side object- Returns:
this
instance.
-
append
Performs a deep comparison of twoObject
arrays.This also will be called for the top level of multi-dimensional, ragged, and multi-typed arrays.
Note that this method does not compare the type of the arrays; it only compares the contents.
- Parameters:
lhs
- the left-hand sideObject[]
rhs
- the right-hand sideObject[]
- Returns:
this
instance.
-
append
Test if twoshort
s are equal.- Parameters:
lhs
- the left-hand sideshort
rhs
- the right-hand sideshort
- Returns:
this
instance.
-
append
Deep comparison of array ofshort
. Length and all values are compared.The method
append(short, short)
is used.- Parameters:
lhs
- the left-hand sideshort[]
rhs
- the right-hand sideshort[]
- Returns:
this
instance.
-
appendSuper
Adds the result ofsuper.equals()
to this builder.- Parameters:
superEquals
- the result of callingsuper.equals()
- Returns:
this
instance.- Since:
- 2.0
-
build
Returnstrue
if the fields that have been checked are all equal. -
isEquals
Returnstrue
if the fields that have been checked are all equal.- Returns:
- boolean
-
reflectionAppend
Tests if twoobjects
by using reflection.It uses
AccessibleObject.setAccessible
to gain access to private fields. This means that it will throw a security exception if run under a security manager, if the permissions are not set up correctly. It is also not as efficient as testing explicitly. Non-primitive fields are compared usingequals()
.If the testTransients field is set to
true
, transient members will be tested, otherwise they are ignored, as they are likely derived fields, and not part of the value of theObject
.Static fields will not be included. Superclass fields will be appended up to and including the specified superclass in field
reflectUpToClass
. A null superclass is treated as java.lang.Object.Field names listed in field
excludeFields
will be ignored.If either class of the compared objects is contained in
bypassReflectionClasses
, both objects are compared by calling the equals method of the left-hand side object with the right-hand side object as an argument.- Parameters:
lhs
- the left-hand side objectrhs
- the right-hand side object- Returns:
this
instance.
-
reset
Reset the EqualsBuilder so you can use the same object again.- Since:
- 2.5
-
setBypassReflectionClasses
SetsClass
es whose instances should be compared by calling theirequals
although being in recursive mode. So the fields of these classes will not be compared recursively by reflection.Here you should name classes having non-transient fields which are cache fields being set lazily.
Prominent example beingString
class with its hash code cache field. Due to the importance of theString
class, it is included in the default bypasses classes. Usually, if you use your own set of classes here, remember to includeString
class, too.- Parameters:
bypassReflectionClasses
- classes to bypass reflection test- Returns:
this
instance.- Since:
- 3.8
- See Also:
-
setEquals
Sets theisEquals
value.- Parameters:
isEquals
- The value to set.- Since:
- 2.1
-
setExcludeFields
Sets field names to be excluded by reflection tests.- Parameters:
excludeFields
- the fields to exclude- Returns:
this
instance.- Since:
- 3.6
-
setReflectUpToClass
Sets the superclass to reflect up to at reflective tests.- Parameters:
reflectUpToClass
- the super class to reflect up to- Returns:
this
instance.- Since:
- 3.6
-
setTestRecursive
Sets whether to test fields recursively, instead of using their equals method, when reflectively comparing objects. String objects, which cache a hash value, are automatically excluded from recursive testing. You may specify other exceptions by callingsetBypassReflectionClasses(List)
.- Parameters:
testRecursive
- whether to do a recursive test- Returns:
this
instance.- Since:
- 3.6
- See Also:
-
setTestTransients
Sets whether to include transient fields when reflectively comparing objects.- Parameters:
testTransients
- whether to test transient fields- Returns:
this
instance.- Since:
- 3.6
-