EqualsBuilder.java

  1. /*
  2.  * Licensed to the Apache Software Foundation (ASF) under one or more
  3.  * contributor license agreements.  See the NOTICE file distributed with
  4.  * this work for additional information regarding copyright ownership.
  5.  * The ASF licenses this file to You under the Apache License, Version 2.0
  6.  * (the "License"); you may not use this file except in compliance with
  7.  * the License.  You may obtain a copy of the License at
  8.  *
  9.  *      http://www.apache.org/licenses/LICENSE-2.0
  10.  *
  11.  * Unless required by applicable law or agreed to in writing, software
  12.  * distributed under the License is distributed on an "AS IS" BASIS,
  13.  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  14.  * See the License for the specific language governing permissions and
  15.  * limitations under the License.
  16.  */
  17. package org.apache.commons.lang3.builder;

  18. import java.lang.reflect.AccessibleObject;
  19. import java.lang.reflect.Field;
  20. import java.lang.reflect.Modifier;
  21. import java.util.ArrayList;
  22. import java.util.Collection;
  23. import java.util.HashSet;
  24. import java.util.List;
  25. import java.util.Set;

  26. import org.apache.commons.lang3.ArrayUtils;
  27. import org.apache.commons.lang3.ClassUtils;
  28. import org.apache.commons.lang3.tuple.Pair;

  29. /**
  30.  * Assists in implementing {@link Object#equals(Object)} methods.
  31.  *
  32.  * <p>This class provides methods to build a good equals method for any
  33.  * class. It follows rules laid out in
  34.  * <a href="https://www.oracle.com/java/technologies/effectivejava.html">Effective Java</a>
  35.  * , by Joshua Bloch. In particular the rule for comparing {@code doubles},
  36.  * {@code floats}, and arrays can be tricky. Also, making sure that
  37.  * {@code equals()} and {@code hashCode()} are consistent can be
  38.  * difficult.</p>
  39.  *
  40.  * <p>Two Objects that compare as equals must generate the same hash code,
  41.  * but two Objects with the same hash code do not have to be equal.</p>
  42.  *
  43.  * <p>All relevant fields should be included in the calculation of equals.
  44.  * Derived fields may be ignored. In particular, any field used in
  45.  * generating a hash code must be used in the equals method, and vice
  46.  * versa.</p>
  47.  *
  48.  * <p>Typical use for the code is as follows:</p>
  49.  * <pre>
  50.  * public boolean equals(Object obj) {
  51.  *   if (obj == null) { return false; }
  52.  *   if (obj == this) { return true; }
  53.  *   if (obj.getClass() != getClass()) {
  54.  *     return false;
  55.  *   }
  56.  *   MyClass rhs = (MyClass) obj;
  57.  *   return new EqualsBuilder()
  58.  *                 .appendSuper(super.equals(obj))
  59.  *                 .append(field1, rhs.field1)
  60.  *                 .append(field2, rhs.field2)
  61.  *                 .append(field3, rhs.field3)
  62.  *                 .isEquals();
  63.  *  }
  64.  * </pre>
  65.  *
  66.  * <p>Alternatively, there is a method that uses reflection to determine
  67.  * the fields to test. Because these fields are usually private, the method,
  68.  * {@code reflectionEquals}, uses {@code AccessibleObject.setAccessible} to
  69.  * change the visibility of the fields. This will fail under a security
  70.  * manager, unless the appropriate permissions are set up correctly. It is
  71.  * also slower than testing explicitly.  Non-primitive fields are compared using
  72.  * {@code equals()}.</p>
  73.  *
  74.  * <p>A typical invocation for this method would look like:</p>
  75.  * <pre>
  76.  * public boolean equals(Object obj) {
  77.  *   return EqualsBuilder.reflectionEquals(this, obj);
  78.  * }
  79.  * </pre>
  80.  *
  81.  * <p>The {@link EqualsExclude} annotation can be used to exclude fields from being
  82.  * used by the {@code reflectionEquals} methods.</p>
  83.  *
  84.  * @since 1.0
  85.  */
  86. public class EqualsBuilder implements Builder<Boolean> {

  87.     /**
  88.      * A registry of objects used by reflection methods to detect cyclical object references and avoid infinite loops.
  89.      *
  90.      * @since 3.0
  91.      */
  92.     private static final ThreadLocal<Set<Pair<IDKey, IDKey>>> REGISTRY = ThreadLocal.withInitial(HashSet::new);

  93.     /*
  94.      * NOTE: we cannot store the actual objects in a HashSet, as that would use the very hashCode()
  95.      * we are in the process of calculating.
  96.      *
  97.      * So we generate a one-to-one mapping from the original object to a new object.
  98.      *
  99.      * Now HashSet uses equals() to determine if two elements with the same hash code really
  100.      * are equal, so we also need to ensure that the replacement objects are only equal
  101.      * if the original objects are identical.
  102.      *
  103.      * The original implementation (2.4 and before) used the System.identityHashCode()
  104.      * method - however this is not guaranteed to generate unique ids (e.g. LANG-459)
  105.      *
  106.      * We now use the IDKey helper class (adapted from org.apache.axis.utils.IDKey)
  107.      * to disambiguate the duplicate ids.
  108.      */

  109.     /**
  110.      * Converters value pair into a register pair.
  111.      *
  112.      * @param lhs {@code this} object
  113.      * @param rhs the other object
  114.      *
  115.      * @return the pair
  116.      */
  117.     static Pair<IDKey, IDKey> getRegisterPair(final Object lhs, final Object rhs) {
  118.         return Pair.of(new IDKey(lhs), new IDKey(rhs));
  119.     }

  120.     /**
  121.      * Returns the registry of object pairs being traversed by the reflection
  122.      * methods in the current thread.
  123.      *
  124.      * @return Set the registry of objects being traversed
  125.      * @since 3.0
  126.      */
  127.     static Set<Pair<IDKey, IDKey>> getRegistry() {
  128.         return REGISTRY.get();
  129.     }

  130.     /**
  131.      * Returns {@code true} if the registry contains the given object pair.
  132.      * Used by the reflection methods to avoid infinite loops.
  133.      * Objects might be swapped therefore a check is needed if the object pair
  134.      * is registered in given or swapped order.
  135.      *
  136.      * @param lhs {@code this} object to lookup in registry
  137.      * @param rhs the other object to lookup on registry
  138.      * @return boolean {@code true} if the registry contains the given object.
  139.      * @since 3.0
  140.      */
  141.     static boolean isRegistered(final Object lhs, final Object rhs) {
  142.         final Set<Pair<IDKey, IDKey>> registry = getRegistry();
  143.         final Pair<IDKey, IDKey> pair = getRegisterPair(lhs, rhs);
  144.         final Pair<IDKey, IDKey> swappedPair = Pair.of(pair.getRight(), pair.getLeft());
  145.         return registry != null && (registry.contains(pair) || registry.contains(swappedPair));
  146.     }

  147.     /**
  148.      * This method uses reflection to determine if the two {@link Object}s
  149.      * are equal.
  150.      *
  151.      * <p>It uses {@code AccessibleObject.setAccessible} to gain access to private
  152.      * fields. This means that it will throw a security exception if run under
  153.      * a security manager, if the permissions are not set up correctly. It is also
  154.      * not as efficient as testing explicitly. Non-primitive fields are compared using
  155.      * {@code equals()}.</p>
  156.      *
  157.      * <p>If the TestTransients parameter is set to {@code true}, transient
  158.      * members will be tested, otherwise they are ignored, as they are likely
  159.      * derived fields, and not part of the value of the {@link Object}.</p>
  160.      *
  161.      * <p>Static fields will not be tested. Superclass fields will be included.</p>
  162.      *
  163.      * @param lhs  {@code this} object
  164.      * @param rhs  the other object
  165.      * @param testTransients  whether to include transient fields
  166.      * @return {@code true} if the two Objects have tested equals.
  167.      *
  168.      * @see EqualsExclude
  169.      */
  170.     public static boolean reflectionEquals(final Object lhs, final Object rhs, final boolean testTransients) {
  171.         return reflectionEquals(lhs, rhs, testTransients, null);
  172.     }

  173.     /**
  174.      * This method uses reflection to determine if the two {@link Object}s
  175.      * are equal.
  176.      *
  177.      * <p>It uses {@code AccessibleObject.setAccessible} to gain access to private
  178.      * fields. This means that it will throw a security exception if run under
  179.      * a security manager, if the permissions are not set up correctly. It is also
  180.      * not as efficient as testing explicitly. Non-primitive fields are compared using
  181.      * {@code equals()}.</p>
  182.      *
  183.      * <p>If the testTransients parameter is set to {@code true}, transient
  184.      * members will be tested, otherwise they are ignored, as they are likely
  185.      * derived fields, and not part of the value of the {@link Object}.</p>
  186.      *
  187.      * <p>Static fields will not be included. Superclass fields will be appended
  188.      * up to and including the specified superclass. A null superclass is treated
  189.      * as java.lang.Object.</p>
  190.      *
  191.      * <p>If the testRecursive parameter is set to {@code true}, non primitive
  192.      * (and non primitive wrapper) field types will be compared by
  193.      * {@link EqualsBuilder} recursively instead of invoking their
  194.      * {@code equals()} method. Leading to a deep reflection equals test.
  195.      *
  196.      * @param lhs  {@code this} object
  197.      * @param rhs  the other object
  198.      * @param testTransients  whether to include transient fields
  199.      * @param reflectUpToClass  the superclass to reflect up to (inclusive),
  200.      *  may be {@code null}
  201.      * @param testRecursive  whether to call reflection equals on non-primitive
  202.      *  fields recursively.
  203.      * @param excludeFields  array of field names to exclude from testing
  204.      * @return {@code true} if the two Objects have tested equals.
  205.      *
  206.      * @see EqualsExclude
  207.      * @since 3.6
  208.      */
  209.     public static boolean reflectionEquals(final Object lhs, final Object rhs, final boolean testTransients, final Class<?> reflectUpToClass,
  210.             final boolean testRecursive, final String... excludeFields) {
  211.         if (lhs == rhs) {
  212.             return true;
  213.         }
  214.         if (lhs == null || rhs == null) {
  215.             return false;
  216.         }
  217.         // @formatter:off
  218.         return new EqualsBuilder()
  219.             .setExcludeFields(excludeFields)
  220.             .setReflectUpToClass(reflectUpToClass)
  221.             .setTestTransients(testTransients)
  222.             .setTestRecursive(testRecursive)
  223.             .reflectionAppend(lhs, rhs)
  224.             .isEquals();
  225.         // @formatter:on
  226.     }

  227.     /**
  228.      * This method uses reflection to determine if the two {@link Object}s
  229.      * are equal.
  230.      *
  231.      * <p>It uses {@code AccessibleObject.setAccessible} to gain access to private
  232.      * fields. This means that it will throw a security exception if run under
  233.      * a security manager, if the permissions are not set up correctly. It is also
  234.      * not as efficient as testing explicitly. Non-primitive fields are compared using
  235.      * {@code equals()}.</p>
  236.      *
  237.      * <p>If the testTransients parameter is set to {@code true}, transient
  238.      * members will be tested, otherwise they are ignored, as they are likely
  239.      * derived fields, and not part of the value of the {@link Object}.</p>
  240.      *
  241.      * <p>Static fields will not be included. Superclass fields will be appended
  242.      * up to and including the specified superclass. A null superclass is treated
  243.      * as java.lang.Object.</p>
  244.      *
  245.      * @param lhs  {@code this} object
  246.      * @param rhs  the other object
  247.      * @param testTransients  whether to include transient fields
  248.      * @param reflectUpToClass  the superclass to reflect up to (inclusive),
  249.      *  may be {@code null}
  250.      * @param excludeFields  array of field names to exclude from testing
  251.      * @return {@code true} if the two Objects have tested equals.
  252.      *
  253.      * @see EqualsExclude
  254.      * @since 2.0
  255.      */
  256.     public static boolean reflectionEquals(final Object lhs, final Object rhs, final boolean testTransients, final Class<?> reflectUpToClass,
  257.             final String... excludeFields) {
  258.         return reflectionEquals(lhs, rhs, testTransients, reflectUpToClass, false, excludeFields);
  259.     }

  260.     /**
  261.      * This method uses reflection to determine if the two {@link Object}s
  262.      * are equal.
  263.      *
  264.      * <p>It uses {@code AccessibleObject.setAccessible} to gain access to private
  265.      * fields. This means that it will throw a security exception if run under
  266.      * a security manager, if the permissions are not set up correctly. It is also
  267.      * not as efficient as testing explicitly. Non-primitive fields are compared using
  268.      * {@code equals()}.</p>
  269.      *
  270.      * <p>Transient members will be not be tested, as they are likely derived
  271.      * fields, and not part of the value of the Object.</p>
  272.      *
  273.      * <p>Static fields will not be tested. Superclass fields will be included.</p>
  274.      *
  275.      * @param lhs  {@code this} object
  276.      * @param rhs  the other object
  277.      * @param excludeFields  Collection of String field names to exclude from testing
  278.      * @return {@code true} if the two Objects have tested equals.
  279.      *
  280.      * @see EqualsExclude
  281.      */
  282.     public static boolean reflectionEquals(final Object lhs, final Object rhs, final Collection<String> excludeFields) {
  283.         return reflectionEquals(lhs, rhs, ReflectionToStringBuilder.toNoNullStringArray(excludeFields));
  284.     }

  285.     /**
  286.      * This method uses reflection to determine if the two {@link Object}s
  287.      * are equal.
  288.      *
  289.      * <p>It uses {@code AccessibleObject.setAccessible} to gain access to private
  290.      * fields. This means that it will throw a security exception if run under
  291.      * a security manager, if the permissions are not set up correctly. It is also
  292.      * not as efficient as testing explicitly. Non-primitive fields are compared using
  293.      * {@code equals()}.</p>
  294.      *
  295.      * <p>Transient members will be not be tested, as they are likely derived
  296.      * fields, and not part of the value of the Object.</p>
  297.      *
  298.      * <p>Static fields will not be tested. Superclass fields will be included.</p>
  299.      *
  300.      * @param lhs  {@code this} object
  301.      * @param rhs  the other object
  302.      * @param excludeFields  array of field names to exclude from testing
  303.      * @return {@code true} if the two Objects have tested equals.
  304.      *
  305.      * @see EqualsExclude
  306.      */
  307.     public static boolean reflectionEquals(final Object lhs, final Object rhs, final String... excludeFields) {
  308.         return reflectionEquals(lhs, rhs, false, null, excludeFields);
  309.     }

  310.     /**
  311.      * Registers the given object pair.
  312.      * Used by the reflection methods to avoid infinite loops.
  313.      *
  314.      * @param lhs {@code this} object to register
  315.      * @param rhs the other object to register
  316.      */
  317.     private static void register(final Object lhs, final Object rhs) {
  318.         getRegistry().add(getRegisterPair(lhs, rhs));
  319.     }

  320.     /**
  321.      * Unregisters the given object pair.
  322.      *
  323.      * <p>
  324.      * Used by the reflection methods to avoid infinite loops.
  325.      * </p>
  326.      *
  327.      * @param lhs {@code this} object to unregister
  328.      * @param rhs the other object to unregister
  329.      * @since 3.0
  330.      */
  331.     private static void unregister(final Object lhs, final Object rhs) {
  332.         final Set<Pair<IDKey, IDKey>> registry = getRegistry();
  333.         registry.remove(getRegisterPair(lhs, rhs));
  334.         if (registry.isEmpty()) {
  335.             REGISTRY.remove();
  336.         }
  337.     }

  338.     /**
  339.      * If the fields tested are equals.
  340.      * The default value is {@code true}.
  341.      */
  342.     private boolean isEquals = true;

  343.     private boolean testTransients;

  344.     private boolean testRecursive;

  345.     private List<Class<?>> bypassReflectionClasses;

  346.     private Class<?> reflectUpToClass;

  347.     private String[] excludeFields;

  348.     /**
  349.      * Constructor for EqualsBuilder.
  350.      *
  351.      * <p>Starts off assuming that equals is {@code true}.</p>
  352.      * @see Object#equals(Object)
  353.      */
  354.     public EqualsBuilder() {
  355.         // set up default classes to bypass reflection for
  356.         bypassReflectionClasses = new ArrayList<>(1);
  357.         bypassReflectionClasses.add(String.class); //hashCode field being lazy but not transient
  358.     }

  359.     /**
  360.      * Test if two {@code booleans}s are equal.
  361.      *
  362.      * @param lhs  the left-hand side {@code boolean}
  363.      * @param rhs  the right-hand side {@code boolean}
  364.      * @return {@code this} instance.
  365.       */
  366.     public EqualsBuilder append(final boolean lhs, final boolean rhs) {
  367.         if (!isEquals) {
  368.             return this;
  369.         }
  370.         isEquals = lhs == rhs;
  371.         return this;
  372.     }

  373.     /**
  374.      * Deep comparison of array of {@code boolean}. Length and all
  375.      * values are compared.
  376.      *
  377.      * <p>The method {@link #append(boolean, boolean)} is used.</p>
  378.      *
  379.      * @param lhs  the left-hand side {@code boolean[]}
  380.      * @param rhs  the right-hand side {@code boolean[]}
  381.      * @return {@code this} instance.
  382.      */
  383.     public EqualsBuilder append(final boolean[] lhs, final boolean[] rhs) {
  384.         if (!isEquals) {
  385.             return this;
  386.         }
  387.         if (lhs == rhs) {
  388.             return this;
  389.         }
  390.         if (lhs == null || rhs == null) {
  391.             this.setEquals(false);
  392.             return this;
  393.         }
  394.         if (lhs.length != rhs.length) {
  395.             this.setEquals(false);
  396.             return this;
  397.         }
  398.         for (int i = 0; i < lhs.length && isEquals; ++i) {
  399.             append(lhs[i], rhs[i]);
  400.         }
  401.         return this;
  402.     }

  403.     /**
  404.      * Test if two {@code byte}s are equal.
  405.      *
  406.      * @param lhs  the left-hand side {@code byte}
  407.      * @param rhs  the right-hand side {@code byte}
  408.      * @return {@code this} instance.
  409.      */
  410.     public EqualsBuilder append(final byte lhs, final byte rhs) {
  411.         if (isEquals) {
  412.             isEquals = lhs == rhs;
  413.         }
  414.         return this;
  415.     }

  416.     /**
  417.      * Deep comparison of array of {@code byte}. Length and all
  418.      * values are compared.
  419.      *
  420.      * <p>The method {@link #append(byte, byte)} is used.</p>
  421.      *
  422.      * @param lhs  the left-hand side {@code byte[]}
  423.      * @param rhs  the right-hand side {@code byte[]}
  424.      * @return {@code this} instance.
  425.      */
  426.     public EqualsBuilder append(final byte[] lhs, final byte[] rhs) {
  427.         if (!isEquals) {
  428.             return this;
  429.         }
  430.         if (lhs == rhs) {
  431.             return this;
  432.         }
  433.         if (lhs == null || rhs == null) {
  434.             setEquals(false);
  435.             return this;
  436.         }
  437.         if (lhs.length != rhs.length) {
  438.             setEquals(false);
  439.             return this;
  440.         }
  441.         for (int i = 0; i < lhs.length && isEquals; ++i) {
  442.             append(lhs[i], rhs[i]);
  443.         }
  444.         return this;
  445.     }

  446.     /**
  447.      * Test if two {@code char}s are equal.
  448.      *
  449.      * @param lhs  the left-hand side {@code char}
  450.      * @param rhs  the right-hand side {@code char}
  451.      * @return {@code this} instance.
  452.      */
  453.     public EqualsBuilder append(final char lhs, final char rhs) {
  454.         if (isEquals) {
  455.             isEquals = lhs == rhs;
  456.         }
  457.         return this;
  458.     }

  459.     /**
  460.      * Deep comparison of array of {@code char}. Length and all
  461.      * values are compared.
  462.      *
  463.      * <p>The method {@link #append(char, char)} is used.</p>
  464.      *
  465.      * @param lhs  the left-hand side {@code char[]}
  466.      * @param rhs  the right-hand side {@code char[]}
  467.      * @return {@code this} instance.
  468.      */
  469.     public EqualsBuilder append(final char[] lhs, final char[] rhs) {
  470.         if (!isEquals) {
  471.             return this;
  472.         }
  473.         if (lhs == rhs) {
  474.             return this;
  475.         }
  476.         if (lhs == null || rhs == null) {
  477.             setEquals(false);
  478.             return this;
  479.         }
  480.         if (lhs.length != rhs.length) {
  481.             setEquals(false);
  482.             return this;
  483.         }
  484.         for (int i = 0; i < lhs.length && isEquals; ++i) {
  485.             append(lhs[i], rhs[i]);
  486.         }
  487.         return this;
  488.     }

  489.     /**
  490.      * Test if two {@code double}s are equal by testing that the
  491.      * pattern of bits returned by {@code doubleToLong} are equal.
  492.      *
  493.      * <p>This handles NaNs, Infinities, and {@code -0.0}.</p>
  494.      *
  495.      * <p>It is compatible with the hash code generated by
  496.      * {@link HashCodeBuilder}.</p>
  497.      *
  498.      * @param lhs  the left-hand side {@code double}
  499.      * @param rhs  the right-hand side {@code double}
  500.      * @return {@code this} instance.
  501.      */
  502.     public EqualsBuilder append(final double lhs, final double rhs) {
  503.         if (isEquals) {
  504.             return append(Double.doubleToLongBits(lhs), Double.doubleToLongBits(rhs));
  505.         }
  506.         return this;
  507.     }

  508.     /**
  509.      * Deep comparison of array of {@code double}. Length and all
  510.      * values are compared.
  511.      *
  512.      * <p>The method {@link #append(double, double)} is used.</p>
  513.      *
  514.      * @param lhs  the left-hand side {@code double[]}
  515.      * @param rhs  the right-hand side {@code double[]}
  516.      * @return {@code this} instance.
  517.      */
  518.     public EqualsBuilder append(final double[] lhs, final double[] rhs) {
  519.         if (!isEquals) {
  520.             return this;
  521.         }
  522.         if (lhs == rhs) {
  523.             return this;
  524.         }
  525.         if (lhs == null || rhs == null) {
  526.             setEquals(false);
  527.             return this;
  528.         }
  529.         if (lhs.length != rhs.length) {
  530.             setEquals(false);
  531.             return this;
  532.         }
  533.         for (int i = 0; i < lhs.length && isEquals; ++i) {
  534.             append(lhs[i], rhs[i]);
  535.         }
  536.         return this;
  537.     }

  538.     /**
  539.      * Test if two {@code float}s are equal by testing that the
  540.      * pattern of bits returned by doubleToLong are equal.
  541.      *
  542.      * <p>This handles NaNs, Infinities, and {@code -0.0}.</p>
  543.      *
  544.      * <p>It is compatible with the hash code generated by
  545.      * {@link HashCodeBuilder}.</p>
  546.      *
  547.      * @param lhs  the left-hand side {@code float}
  548.      * @param rhs  the right-hand side {@code float}
  549.      * @return {@code this} instance.
  550.      */
  551.     public EqualsBuilder append(final float lhs, final float rhs) {
  552.         if (isEquals) {
  553.             return append(Float.floatToIntBits(lhs), Float.floatToIntBits(rhs));
  554.         }
  555.         return this;
  556.     }

  557.     /**
  558.      * Deep comparison of array of {@code float}. Length and all
  559.      * values are compared.
  560.      *
  561.      * <p>The method {@link #append(float, float)} is used.</p>
  562.      *
  563.      * @param lhs  the left-hand side {@code float[]}
  564.      * @param rhs  the right-hand side {@code float[]}
  565.      * @return {@code this} instance.
  566.      */
  567.     public EqualsBuilder append(final float[] lhs, final float[] rhs) {
  568.         if (!isEquals) {
  569.             return this;
  570.         }
  571.         if (lhs == rhs) {
  572.             return this;
  573.         }
  574.         if (lhs == null || rhs == null) {
  575.             setEquals(false);
  576.             return this;
  577.         }
  578.         if (lhs.length != rhs.length) {
  579.             setEquals(false);
  580.             return this;
  581.         }
  582.         for (int i = 0; i < lhs.length && isEquals; ++i) {
  583.             append(lhs[i], rhs[i]);
  584.         }
  585.         return this;
  586.     }

  587.     /**
  588.      * Test if two {@code int}s are equal.
  589.      *
  590.      * @param lhs  the left-hand side {@code int}
  591.      * @param rhs  the right-hand side {@code int}
  592.      * @return {@code this} instance.
  593.      */
  594.     public EqualsBuilder append(final int lhs, final int rhs) {
  595.         if (isEquals) {
  596.             isEquals = lhs == rhs;
  597.         }
  598.         return this;
  599.     }

  600.     /**
  601.      * Deep comparison of array of {@code int}. Length and all
  602.      * values are compared.
  603.      *
  604.      * <p>The method {@link #append(int, int)} is used.</p>
  605.      *
  606.      * @param lhs  the left-hand side {@code int[]}
  607.      * @param rhs  the right-hand side {@code int[]}
  608.      * @return {@code this} instance.
  609.      */
  610.     public EqualsBuilder append(final int[] lhs, final int[] rhs) {
  611.         if (!isEquals) {
  612.             return this;
  613.         }
  614.         if (lhs == rhs) {
  615.             return this;
  616.         }
  617.         if (lhs == null || rhs == null) {
  618.             setEquals(false);
  619.             return this;
  620.         }
  621.         if (lhs.length != rhs.length) {
  622.             setEquals(false);
  623.             return this;
  624.         }
  625.         for (int i = 0; i < lhs.length && isEquals; ++i) {
  626.             append(lhs[i], rhs[i]);
  627.         }
  628.         return this;
  629.     }

  630.     /**
  631.      * Test if two {@code long}s are equal.
  632.      *
  633.      * @param lhs
  634.      *                  the left-hand side {@code long}
  635.      * @param rhs
  636.      *                  the right-hand side {@code long}
  637.      * @return {@code this} instance.
  638.      */
  639.     public EqualsBuilder append(final long lhs, final long rhs) {
  640.         if (isEquals) {
  641.             isEquals = lhs == rhs;
  642.         }
  643.         return this;
  644.     }

  645.     /**
  646.      * Deep comparison of array of {@code long}. Length and all
  647.      * values are compared.
  648.      *
  649.      * <p>The method {@link #append(long, long)} is used.</p>
  650.      *
  651.      * @param lhs  the left-hand side {@code long[]}
  652.      * @param rhs  the right-hand side {@code long[]}
  653.      * @return {@code this} instance.
  654.      */
  655.     public EqualsBuilder append(final long[] lhs, final long[] rhs) {
  656.         if (!isEquals) {
  657.             return this;
  658.         }
  659.         if (lhs == rhs) {
  660.             return this;
  661.         }
  662.         if (lhs == null || rhs == null) {
  663.             setEquals(false);
  664.             return this;
  665.         }
  666.         if (lhs.length != rhs.length) {
  667.             setEquals(false);
  668.             return this;
  669.         }
  670.         for (int i = 0; i < lhs.length && isEquals; ++i) {
  671.             append(lhs[i], rhs[i]);
  672.         }
  673.         return this;
  674.     }

  675.     /**
  676.      * Test if two {@link Object}s are equal using either
  677.      * #{@link #reflectionAppend(Object, Object)}, if object are non
  678.      * primitives (or wrapper of primitives) or if field {@code testRecursive}
  679.      * is set to {@code false}. Otherwise, using their
  680.      * {@code equals} method.
  681.      *
  682.      * @param lhs  the left-hand side object
  683.      * @param rhs  the right-hand side object
  684.      * @return {@code this} instance.
  685.      */
  686.     public EqualsBuilder append(final Object lhs, final Object rhs) {
  687.         if (!isEquals) {
  688.             return this;
  689.         }
  690.         if (lhs == rhs) {
  691.             return this;
  692.         }
  693.         if (lhs == null || rhs == null) {
  694.             setEquals(false);
  695.             return this;
  696.         }
  697.         final Class<?> lhsClass = lhs.getClass();
  698.         if (lhsClass.isArray()) {
  699.             // factor out array case in order to keep method small enough
  700.             // to be inlined
  701.             appendArray(lhs, rhs);
  702.         } else // The simple case, not an array, just test the element
  703.         if (testRecursive && !ClassUtils.isPrimitiveOrWrapper(lhsClass)) {
  704.             reflectionAppend(lhs, rhs);
  705.         } else {
  706.             isEquals = lhs.equals(rhs);
  707.         }
  708.         return this;
  709.     }

  710.     /**
  711.      * Performs a deep comparison of two {@link Object} arrays.
  712.      *
  713.      * <p>This also will be called for the top level of
  714.      * multi-dimensional, ragged, and multi-typed arrays.</p>
  715.      *
  716.      * <p>Note that this method does not compare the type of the arrays; it only
  717.      * compares the contents.</p>
  718.      *
  719.      * @param lhs  the left-hand side {@code Object[]}
  720.      * @param rhs  the right-hand side {@code Object[]}
  721.      * @return {@code this} instance.
  722.      */
  723.     public EqualsBuilder append(final Object[] lhs, final Object[] rhs) {
  724.         if (!isEquals) {
  725.             return this;
  726.         }
  727.         if (lhs == rhs) {
  728.             return this;
  729.         }
  730.         if (lhs == null || rhs == null) {
  731.             setEquals(false);
  732.             return this;
  733.         }
  734.         if (lhs.length != rhs.length) {
  735.             setEquals(false);
  736.             return this;
  737.         }
  738.         for (int i = 0; i < lhs.length && isEquals; ++i) {
  739.             append(lhs[i], rhs[i]);
  740.         }
  741.         return this;
  742.     }

  743.     /**
  744.      * Test if two {@code short}s are equal.
  745.      *
  746.      * @param lhs  the left-hand side {@code short}
  747.      * @param rhs  the right-hand side {@code short}
  748.      * @return {@code this} instance.
  749.      */
  750.     public EqualsBuilder append(final short lhs, final short rhs) {
  751.         if (isEquals) {
  752.             isEquals = lhs == rhs;
  753.         }
  754.         return this;
  755.     }

  756.     /**
  757.      * Deep comparison of array of {@code short}. Length and all
  758.      * values are compared.
  759.      *
  760.      * <p>The method {@link #append(short, short)} is used.</p>
  761.      *
  762.      * @param lhs  the left-hand side {@code short[]}
  763.      * @param rhs  the right-hand side {@code short[]}
  764.      * @return {@code this} instance.
  765.      */
  766.     public EqualsBuilder append(final short[] lhs, final short[] rhs) {
  767.         if (!isEquals) {
  768.             return this;
  769.         }
  770.         if (lhs == rhs) {
  771.             return this;
  772.         }
  773.         if (lhs == null || rhs == null) {
  774.             setEquals(false);
  775.             return this;
  776.         }
  777.         if (lhs.length != rhs.length) {
  778.             setEquals(false);
  779.             return this;
  780.         }
  781.         for (int i = 0; i < lhs.length && isEquals; ++i) {
  782.             append(lhs[i], rhs[i]);
  783.         }
  784.         return this;
  785.     }

  786.     /**
  787.      * Test if an {@link Object} is equal to an array.
  788.      *
  789.      * @param lhs  the left-hand side object, an array
  790.      * @param rhs  the right-hand side object
  791.      */
  792.     private void appendArray(final Object lhs, final Object rhs) {
  793.         // First we compare different dimensions, for example: a boolean[][] to a boolean[]
  794.         // then we 'Switch' on type of array, to dispatch to the correct handler
  795.         // This handles multidimensional arrays of the same depth
  796.         if (lhs.getClass() != rhs.getClass()) {
  797.             setEquals(false);
  798.         } else if (lhs instanceof long[]) {
  799.             append((long[]) lhs, (long[]) rhs);
  800.         } else if (lhs instanceof int[]) {
  801.             append((int[]) lhs, (int[]) rhs);
  802.         } else if (lhs instanceof short[]) {
  803.             append((short[]) lhs, (short[]) rhs);
  804.         } else if (lhs instanceof char[]) {
  805.             append((char[]) lhs, (char[]) rhs);
  806.         } else if (lhs instanceof byte[]) {
  807.             append((byte[]) lhs, (byte[]) rhs);
  808.         } else if (lhs instanceof double[]) {
  809.             append((double[]) lhs, (double[]) rhs);
  810.         } else if (lhs instanceof float[]) {
  811.             append((float[]) lhs, (float[]) rhs);
  812.         } else if (lhs instanceof boolean[]) {
  813.             append((boolean[]) lhs, (boolean[]) rhs);
  814.         } else {
  815.             // Not an array of primitives
  816.             append((Object[]) lhs, (Object[]) rhs);
  817.         }
  818.     }

  819.     /**
  820.      * Adds the result of {@code super.equals()} to this builder.
  821.      *
  822.      * @param superEquals  the result of calling {@code super.equals()}
  823.      * @return {@code this} instance.
  824.      * @since 2.0
  825.      */
  826.     public EqualsBuilder appendSuper(final boolean superEquals) {
  827.         if (!isEquals) {
  828.             return this;
  829.         }
  830.         isEquals = superEquals;
  831.         return this;
  832.     }

  833.     /**
  834.      * Returns {@code true} if the fields that have been checked
  835.      * are all equal.
  836.      *
  837.      * @return {@code true} if all of the fields that have been checked
  838.      *         are equal, {@code false} otherwise.
  839.      *
  840.      * @since 3.0
  841.      */
  842.     @Override
  843.     public Boolean build() {
  844.         return Boolean.valueOf(isEquals());
  845.     }

  846.     /**
  847.      * Returns {@code true} if the fields that have been checked
  848.      * are all equal.
  849.      *
  850.      * @return boolean
  851.      */
  852.     public boolean isEquals() {
  853.         return isEquals;
  854.     }

  855.     /**
  856.      * Tests if two {@code objects} by using reflection.
  857.      *
  858.      * <p>It uses {@code AccessibleObject.setAccessible} to gain access to private
  859.      * fields. This means that it will throw a security exception if run under
  860.      * a security manager, if the permissions are not set up correctly. It is also
  861.      * not as efficient as testing explicitly. Non-primitive fields are compared using
  862.      * {@code equals()}.</p>
  863.      *
  864.      * <p>If the testTransients field is set to {@code true}, transient
  865.      * members will be tested, otherwise they are ignored, as they are likely
  866.      * derived fields, and not part of the value of the {@link Object}.</p>
  867.      *
  868.      * <p>Static fields will not be included. Superclass fields will be appended
  869.      * up to and including the specified superclass in field {@code reflectUpToClass}.
  870.      * A null superclass is treated as java.lang.Object.</p>
  871.      *
  872.      * <p>Field names listed in field {@code excludeFields} will be ignored.</p>
  873.      *
  874.      * <p>If either class of the compared objects is contained in
  875.      * {@code bypassReflectionClasses}, both objects are compared by calling
  876.      * the equals method of the left-hand side object with the right-hand side object as an argument.</p>
  877.      *
  878.      * @param lhs  the left-hand side object
  879.      * @param rhs  the right-hand side object
  880.      * @return {@code this} instance.
  881.      */
  882.     public EqualsBuilder reflectionAppend(final Object lhs, final Object rhs) {
  883.         if (!isEquals) {
  884.             return this;
  885.         }
  886.         if (lhs == rhs) {
  887.             return this;
  888.         }
  889.         if (lhs == null || rhs == null) {
  890.             isEquals = false;
  891.             return this;
  892.         }

  893.         // Find the leaf class since there may be transients in the leaf
  894.         // class or in classes between the leaf and root.
  895.         // If we are not testing transients or a subclass has no ivars,
  896.         // then a subclass can test equals to a superclass.
  897.         final Class<?> lhsClass = lhs.getClass();
  898.         final Class<?> rhsClass = rhs.getClass();
  899.         Class<?> testClass;
  900.         if (lhsClass.isInstance(rhs)) {
  901.             testClass = lhsClass;
  902.             if (!rhsClass.isInstance(lhs)) {
  903.                 // rhsClass is a subclass of lhsClass
  904.                 testClass = rhsClass;
  905.             }
  906.         } else if (rhsClass.isInstance(lhs)) {
  907.             testClass = rhsClass;
  908.             if (!lhsClass.isInstance(rhs)) {
  909.                 // lhsClass is a subclass of rhsClass
  910.                 testClass = lhsClass;
  911.             }
  912.         } else {
  913.             // The two classes are not related.
  914.             isEquals = false;
  915.             return this;
  916.         }

  917.         try {
  918.             if (testClass.isArray()) {
  919.                 append(lhs, rhs);
  920.             } else //If either class is being excluded, call normal object equals method on lhsClass.
  921.             if (bypassReflectionClasses != null
  922.                     && (bypassReflectionClasses.contains(lhsClass) || bypassReflectionClasses.contains(rhsClass))) {
  923.                 isEquals = lhs.equals(rhs);
  924.             } else {
  925.                 reflectionAppend(lhs, rhs, testClass);
  926.                 while (testClass.getSuperclass() != null && testClass != reflectUpToClass) {
  927.                     testClass = testClass.getSuperclass();
  928.                     reflectionAppend(lhs, rhs, testClass);
  929.                 }
  930.             }
  931.         } catch (final IllegalArgumentException e) {
  932.             // In this case, we tried to test a subclass vs. a superclass and
  933.             // the subclass has ivars or the ivars are transient and
  934.             // we are testing transients.
  935.             // If a subclass has ivars that we are trying to test them, we get an
  936.             // exception and we know that the objects are not equal.
  937.             isEquals = false;
  938.         }
  939.         return this;
  940.     }

  941.     /**
  942.      * Appends the fields and values defined by the given object of the
  943.      * given Class.
  944.      *
  945.      * @param lhs  the left-hand side object
  946.      * @param rhs  the right-hand side object
  947.      * @param clazz  the class to append details of
  948.      */
  949.     private void reflectionAppend(
  950.         final Object lhs,
  951.         final Object rhs,
  952.         final Class<?> clazz) {

  953.         if (isRegistered(lhs, rhs)) {
  954.             return;
  955.         }

  956.         try {
  957.             register(lhs, rhs);
  958.             final Field[] fields = clazz.getDeclaredFields();
  959.             AccessibleObject.setAccessible(fields, true);
  960.             for (int i = 0; i < fields.length && isEquals; i++) {
  961.                 final Field field = fields[i];
  962.                 if (!ArrayUtils.contains(excludeFields, field.getName())
  963.                     && !field.getName().contains("$")
  964.                     && (testTransients || !Modifier.isTransient(field.getModifiers()))
  965.                     && !Modifier.isStatic(field.getModifiers())
  966.                     && !field.isAnnotationPresent(EqualsExclude.class)) {
  967.                     append(Reflection.getUnchecked(field, lhs), Reflection.getUnchecked(field, rhs));
  968.                 }
  969.             }
  970.         } finally {
  971.             unregister(lhs, rhs);
  972.         }
  973.     }

  974.     /**
  975.      * Reset the EqualsBuilder so you can use the same object again.
  976.      *
  977.      * @since 2.5
  978.      */
  979.     public void reset() {
  980.         isEquals = true;
  981.     }

  982.     /**
  983.      * Sets {@link Class}es whose instances should be compared by calling their {@code equals}
  984.      * although being in recursive mode. So the fields of these classes will not be compared recursively by reflection.
  985.      *
  986.      * <p>Here you should name classes having non-transient fields which are cache fields being set lazily.<br>
  987.      * Prominent example being {@link String} class with its hash code cache field. Due to the importance
  988.      * of the {@link String} class, it is included in the default bypasses classes. Usually, if you use
  989.      * your own set of classes here, remember to include {@link String} class, too.</p>
  990.      *
  991.      * @param bypassReflectionClasses  classes to bypass reflection test
  992.      * @return {@code this} instance.
  993.      * @see #setTestRecursive(boolean)
  994.      * @since 3.8
  995.      */
  996.     public EqualsBuilder setBypassReflectionClasses(final List<Class<?>> bypassReflectionClasses) {
  997.         this.bypassReflectionClasses = bypassReflectionClasses;
  998.         return this;
  999.     }

  1000.     /**
  1001.      * Sets the {@code isEquals} value.
  1002.      *
  1003.      * @param isEquals The value to set.
  1004.      * @since 2.1
  1005.      */
  1006.     protected void setEquals(final boolean isEquals) {
  1007.         this.isEquals = isEquals;
  1008.     }

  1009.     /**
  1010.      * Sets field names to be excluded by reflection tests.
  1011.      *
  1012.      * @param excludeFields the fields to exclude
  1013.      * @return {@code this} instance.
  1014.      * @since 3.6
  1015.      */
  1016.     public EqualsBuilder setExcludeFields(final String... excludeFields) {
  1017.         this.excludeFields = excludeFields;
  1018.         return this;
  1019.     }

  1020.     /**
  1021.      * Sets the superclass to reflect up to at reflective tests.
  1022.      *
  1023.      * @param reflectUpToClass the super class to reflect up to
  1024.      * @return {@code this} instance.
  1025.      * @since 3.6
  1026.      */
  1027.     public EqualsBuilder setReflectUpToClass(final Class<?> reflectUpToClass) {
  1028.         this.reflectUpToClass = reflectUpToClass;
  1029.         return this;
  1030.     }

  1031.     /**
  1032.      * Sets whether to test fields recursively, instead of using their equals method, when reflectively comparing objects.
  1033.      * String objects, which cache a hash value, are automatically excluded from recursive testing.
  1034.      * You may specify other exceptions by calling {@link #setBypassReflectionClasses(List)}.
  1035.      *
  1036.      * @param testRecursive whether to do a recursive test
  1037.      * @return {@code this} instance.
  1038.      * @see #setBypassReflectionClasses(List)
  1039.      * @since 3.6
  1040.      */
  1041.     public EqualsBuilder setTestRecursive(final boolean testRecursive) {
  1042.         this.testRecursive = testRecursive;
  1043.         return this;
  1044.     }

  1045.     /**
  1046.      * Sets whether to include transient fields when reflectively comparing objects.
  1047.      *
  1048.      * @param testTransients whether to test transient fields
  1049.      * @return {@code this} instance.
  1050.      * @since 3.6
  1051.      */
  1052.     public EqualsBuilder setTestTransients(final boolean testTransients) {
  1053.         this.testTransients = testTransients;
  1054.         return this;
  1055.     }
  1056. }