CompareToBuilder.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.Collection;
  22. import java.util.Comparator;
  23. import java.util.Objects;

  24. import org.apache.commons.lang3.ArrayUtils;
  25. import org.apache.commons.lang3.ObjectUtils;

  26. /**
  27.  * Assists in implementing {@link Comparable#compareTo(Object)} methods.
  28.  *
  29.  * <p>It is consistent with {@code equals(Object)} and
  30.  * {@code hashCode()} built with {@link EqualsBuilder} and
  31.  * {@link HashCodeBuilder}.</p>
  32.  *
  33.  * <p>Two Objects that compare equal using {@code equals(Object)} should normally
  34.  * also compare equal using {@code compareTo(Object)}.</p>
  35.  *
  36.  * <p>All relevant fields should be included in the calculation of the
  37.  * comparison. Derived fields may be ignored. The same fields, in the same
  38.  * order, should be used in both {@code compareTo(Object)} and
  39.  * {@code equals(Object)}.</p>
  40.  *
  41.  * <p>To use this class write code as follows:</p>
  42.  *
  43.  * <pre>
  44.  * public class MyClass {
  45.  *   String field1;
  46.  *   int field2;
  47.  *   boolean field3;
  48.  *
  49.  *   ...
  50.  *
  51.  *   public int compareTo(Object o) {
  52.  *     MyClass myClass = (MyClass) o;
  53.  *     return new CompareToBuilder()
  54.  *       .appendSuper(super.compareTo(o)
  55.  *       .append(this.field1, myClass.field1)
  56.  *       .append(this.field2, myClass.field2)
  57.  *       .append(this.field3, myClass.field3)
  58.  *       .toComparison();
  59.  *   }
  60.  * }
  61.  * </pre>
  62.  *
  63.  * <p>Values are compared in the order they are appended to the builder. If any comparison returns
  64.  * a non-zero result, then that value will be the result returned by {@code toComparison()} and all
  65.  * subsequent comparisons are skipped.</p>
  66.  *
  67.  * <p>Alternatively, there are {@link #reflectionCompare(Object, Object) reflectionCompare} methods that use
  68.  * reflection to determine the fields to append. Because fields can be private,
  69.  * {@code reflectionCompare} uses {@link java.lang.reflect.AccessibleObject#setAccessible(boolean)} to
  70.  * bypass normal access control checks. This will fail under a security manager,
  71.  * unless the appropriate permissions are set up correctly. It is also
  72.  * slower than appending explicitly.</p>
  73.  *
  74.  * <p>A typical implementation of {@code compareTo(Object)} using
  75.  * {@code reflectionCompare} looks like:</p>

  76.  * <pre>
  77.  * public int compareTo(Object o) {
  78.  *   return CompareToBuilder.reflectionCompare(this, o);
  79.  * }
  80.  * </pre>
  81.  *
  82.  * <p>The reflective methods compare object fields in the order returned by
  83.  * {@link Class#getDeclaredFields()}. The fields of the class are compared first, followed by those
  84.  * of its parent classes (in order from the bottom to the top of the class hierarchy).</p>
  85.  *
  86.  * @see Comparable
  87.  * @see Object#equals(Object)
  88.  * @see Object#hashCode()
  89.  * @see EqualsBuilder
  90.  * @see HashCodeBuilder
  91.  * @since 1.0
  92.  */
  93. public class CompareToBuilder implements Builder<Integer> {

  94.     /**
  95.      * Appends to {@code builder} the comparison of {@code lhs}
  96.      * to {@code rhs} using the fields defined in {@code clazz}.
  97.      *
  98.      * @param lhs  left-hand side object
  99.      * @param rhs  right-hand side object
  100.      * @param clazz  {@link Class} that defines fields to be compared
  101.      * @param builder  {@link CompareToBuilder} to append to
  102.      * @param useTransients  whether to compare transient fields
  103.      * @param excludeFields  fields to exclude
  104.      */
  105.     private static void reflectionAppend(
  106.         final Object lhs,
  107.         final Object rhs,
  108.         final Class<?> clazz,
  109.         final CompareToBuilder builder,
  110.         final boolean useTransients,
  111.         final String[] excludeFields) {

  112.         final Field[] fields = clazz.getDeclaredFields();
  113.         AccessibleObject.setAccessible(fields, true);
  114.         for (int i = 0; i < fields.length && builder.comparison == 0; i++) {
  115.             final Field field = fields[i];
  116.             if (!ArrayUtils.contains(excludeFields, field.getName())
  117.                 && !field.getName().contains("$")
  118.                 && (useTransients || !Modifier.isTransient(field.getModifiers()))
  119.                 && !Modifier.isStatic(field.getModifiers())) {
  120.                 // IllegalAccessException can't happen. Would get a Security exception instead.
  121.                 // Throw a runtime exception in case the impossible happens.
  122.                 builder.append(Reflection.getUnchecked(field, lhs), Reflection.getUnchecked(field, rhs));
  123.             }
  124.         }
  125.     }

  126.     /**
  127.      * Compares two {@link Object}s via reflection.
  128.      *
  129.      * <p>Fields can be private, thus {@code AccessibleObject.setAccessible}
  130.      * is used to bypass normal access control checks. This will fail under a
  131.      * security manager unless the appropriate permissions are set.</p>
  132.      *
  133.      * <ul>
  134.      * <li>Static fields will not be compared</li>
  135.      * <li>Transient members will be not be compared, as they are likely derived
  136.      *     fields</li>
  137.      * <li>Superclass fields will be compared</li>
  138.      * </ul>
  139.      *
  140.      * <p>If both {@code lhs} and {@code rhs} are {@code null},
  141.      * they are considered equal.</p>
  142.      *
  143.      * @param lhs  left-hand side object
  144.      * @param rhs  right-hand side object
  145.      * @return a negative integer, zero, or a positive integer as {@code lhs}
  146.      *  is less than, equal to, or greater than {@code rhs}
  147.      * @throws NullPointerException  if either (but not both) parameters are
  148.      *  {@code null}
  149.      * @throws ClassCastException  if {@code rhs} is not assignment-compatible
  150.      *  with {@code lhs}
  151.      */
  152.     public static int reflectionCompare(final Object lhs, final Object rhs) {
  153.         return reflectionCompare(lhs, rhs, false, null);
  154.     }

  155.     /**
  156.      * Compares two {@link Object}s via reflection.
  157.      *
  158.      * <p>Fields can be private, thus {@code AccessibleObject.setAccessible}
  159.      * is used to bypass normal access control checks. This will fail under a
  160.      * security manager unless the appropriate permissions are set.</p>
  161.      *
  162.      * <ul>
  163.      * <li>Static fields will not be compared</li>
  164.      * <li>If {@code compareTransients} is {@code true},
  165.      *     compares transient members.  Otherwise ignores them, as they
  166.      *     are likely derived fields.</li>
  167.      * <li>Superclass fields will be compared</li>
  168.      * </ul>
  169.      *
  170.      * <p>If both {@code lhs} and {@code rhs} are {@code null},
  171.      * they are considered equal.</p>
  172.      *
  173.      * @param lhs  left-hand side object
  174.      * @param rhs  right-hand side object
  175.      * @param compareTransients  whether to compare transient fields
  176.      * @return a negative integer, zero, or a positive integer as {@code lhs}
  177.      *  is less than, equal to, or greater than {@code rhs}
  178.      * @throws NullPointerException  if either {@code lhs} or {@code rhs}
  179.      *  (but not both) is {@code null}
  180.      * @throws ClassCastException  if {@code rhs} is not assignment-compatible
  181.      *  with {@code lhs}
  182.      */
  183.     public static int reflectionCompare(final Object lhs, final Object rhs, final boolean compareTransients) {
  184.         return reflectionCompare(lhs, rhs, compareTransients, null);
  185.     }

  186.     /**
  187.      * Compares two {@link Object}s via reflection.
  188.      *
  189.      * <p>Fields can be private, thus {@code AccessibleObject.setAccessible}
  190.      * is used to bypass normal access control checks. This will fail under a
  191.      * security manager unless the appropriate permissions are set.</p>
  192.      *
  193.      * <ul>
  194.      * <li>Static fields will not be compared</li>
  195.      * <li>If the {@code compareTransients} is {@code true},
  196.      *     compares transient members.  Otherwise ignores them, as they
  197.      *     are likely derived fields.</li>
  198.      * <li>Compares superclass fields up to and including {@code reflectUpToClass}.
  199.      *     If {@code reflectUpToClass} is {@code null}, compares all superclass fields.</li>
  200.      * </ul>
  201.      *
  202.      * <p>If both {@code lhs} and {@code rhs} are {@code null},
  203.      * they are considered equal.</p>
  204.      *
  205.      * @param lhs  left-hand side object
  206.      * @param rhs  right-hand side object
  207.      * @param compareTransients  whether to compare transient fields
  208.      * @param reflectUpToClass  last superclass for which fields are compared
  209.      * @param excludeFields  fields to exclude
  210.      * @return a negative integer, zero, or a positive integer as {@code lhs}
  211.      *  is less than, equal to, or greater than {@code rhs}
  212.      * @throws NullPointerException  if either {@code lhs} or {@code rhs}
  213.      *  (but not both) is {@code null}
  214.      * @throws ClassCastException  if {@code rhs} is not assignment-compatible
  215.      *  with {@code lhs}
  216.      * @since 2.2 (2.0 as {@code reflectionCompare(Object, Object, boolean, Class)})
  217.      */
  218.     public static int reflectionCompare(
  219.         final Object lhs,
  220.         final Object rhs,
  221.         final boolean compareTransients,
  222.         final Class<?> reflectUpToClass,
  223.         final String... excludeFields) {

  224.         if (lhs == rhs) {
  225.             return 0;
  226.         }
  227.         Objects.requireNonNull(lhs, "lhs");
  228.         Objects.requireNonNull(rhs, "rhs");

  229.         Class<?> lhsClazz = lhs.getClass();
  230.         if (!lhsClazz.isInstance(rhs)) {
  231.             throw new ClassCastException();
  232.         }
  233.         final CompareToBuilder compareToBuilder = new CompareToBuilder();
  234.         reflectionAppend(lhs, rhs, lhsClazz, compareToBuilder, compareTransients, excludeFields);
  235.         while (lhsClazz.getSuperclass() != null && lhsClazz != reflectUpToClass) {
  236.             lhsClazz = lhsClazz.getSuperclass();
  237.             reflectionAppend(lhs, rhs, lhsClazz, compareToBuilder, compareTransients, excludeFields);
  238.         }
  239.         return compareToBuilder.toComparison();
  240.     }

  241.     /**
  242.      * Compares two {@link Object}s via reflection.
  243.      *
  244.      * <p>Fields can be private, thus {@code AccessibleObject.setAccessible}
  245.      * is used to bypass normal access control checks. This will fail under a
  246.      * security manager unless the appropriate permissions are set.</p>
  247.      *
  248.      * <ul>
  249.      * <li>Static fields will not be compared</li>
  250.      * <li>If {@code compareTransients} is {@code true},
  251.      *     compares transient members.  Otherwise ignores them, as they
  252.      *     are likely derived fields.</li>
  253.      * <li>Superclass fields will be compared</li>
  254.      * </ul>
  255.      *
  256.      * <p>If both {@code lhs} and {@code rhs} are {@code null},
  257.      * they are considered equal.</p>
  258.      *
  259.      * @param lhs  left-hand side object
  260.      * @param rhs  right-hand side object
  261.      * @param excludeFields  Collection of String fields to exclude
  262.      * @return a negative integer, zero, or a positive integer as {@code lhs}
  263.      *  is less than, equal to, or greater than {@code rhs}
  264.      * @throws NullPointerException  if either {@code lhs} or {@code rhs}
  265.      *  (but not both) is {@code null}
  266.      * @throws ClassCastException  if {@code rhs} is not assignment-compatible
  267.      *  with {@code lhs}
  268.      * @since 2.2
  269.      */
  270.     public static int reflectionCompare(final Object lhs, final Object rhs, final Collection<String> excludeFields) {
  271.         return reflectionCompare(lhs, rhs, ReflectionToStringBuilder.toNoNullStringArray(excludeFields));
  272.     }

  273.     /**
  274.      * Compares two {@link Object}s via reflection.
  275.      *
  276.      * <p>Fields can be private, thus {@code AccessibleObject.setAccessible}
  277.      * is used to bypass normal access control checks. This will fail under a
  278.      * security manager unless the appropriate permissions are set.</p>
  279.      *
  280.      * <ul>
  281.      * <li>Static fields will not be compared</li>
  282.      * <li>If {@code compareTransients} is {@code true},
  283.      *     compares transient members.  Otherwise ignores them, as they
  284.      *     are likely derived fields.</li>
  285.      * <li>Superclass fields will be compared</li>
  286.      * </ul>
  287.      *
  288.      * <p>If both {@code lhs} and {@code rhs} are {@code null},
  289.      * they are considered equal.</p>
  290.      *
  291.      * @param lhs  left-hand side object
  292.      * @param rhs  right-hand side object
  293.      * @param excludeFields  array of fields to exclude
  294.      * @return a negative integer, zero, or a positive integer as {@code lhs}
  295.      *  is less than, equal to, or greater than {@code rhs}
  296.      * @throws NullPointerException  if either {@code lhs} or {@code rhs}
  297.      *  (but not both) is {@code null}
  298.      * @throws ClassCastException  if {@code rhs} is not assignment-compatible
  299.      *  with {@code lhs}
  300.      * @since 2.2
  301.      */
  302.     public static int reflectionCompare(final Object lhs, final Object rhs, final String... excludeFields) {
  303.         return reflectionCompare(lhs, rhs, false, null, excludeFields);
  304.     }

  305.     /**
  306.      * Current state of the comparison as appended fields are checked.
  307.      */
  308.     private int comparison;

  309.     /**
  310.      * Constructor for CompareToBuilder.
  311.      *
  312.      * <p>Starts off assuming that the objects are equal. Multiple calls are
  313.      * then made to the various append methods, followed by a call to
  314.      * {@link #toComparison} to get the result.</p>
  315.      */
  316.     public CompareToBuilder() {
  317.         comparison = 0;
  318.     }

  319.     /**
  320.      * Appends to the {@code builder} the comparison of
  321.      * two {@code booleans}s.
  322.      *
  323.      * @param lhs  left-hand side value
  324.      * @param rhs  right-hand side value
  325.      * @return {@code this} instance.
  326.       */
  327.     public CompareToBuilder append(final boolean lhs, final boolean rhs) {
  328.         if (comparison != 0) {
  329.             return this;
  330.         }
  331.         if (lhs == rhs) {
  332.             return this;
  333.         }
  334.         if (lhs) {
  335.             comparison = 1;
  336.         } else {
  337.             comparison = -1;
  338.         }
  339.         return this;
  340.     }

  341.     /**
  342.      * Appends to the {@code builder} the deep comparison of
  343.      * two {@code boolean} arrays.
  344.      *
  345.      * <ol>
  346.      *  <li>Check if arrays are the same using {@code ==}</li>
  347.      *  <li>Check if for {@code null}, {@code null} is less than non-{@code null}</li>
  348.      *  <li>Check array length, a shorter length array is less than a longer length array</li>
  349.      *  <li>Check array contents element by element using {@link #append(boolean, boolean)}</li>
  350.      * </ol>
  351.      *
  352.      * @param lhs  left-hand side array
  353.      * @param rhs  right-hand side array
  354.      * @return {@code this} instance.
  355.      */
  356.     public CompareToBuilder append(final boolean[] lhs, final boolean[] rhs) {
  357.         if (comparison != 0) {
  358.             return this;
  359.         }
  360.         if (lhs == rhs) {
  361.             return this;
  362.         }
  363.         if (lhs == null) {
  364.             comparison = -1;
  365.             return this;
  366.         }
  367.         if (rhs == null) {
  368.             comparison = 1;
  369.             return this;
  370.         }
  371.         if (lhs.length != rhs.length) {
  372.             comparison = lhs.length < rhs.length ? -1 : 1;
  373.             return this;
  374.         }
  375.         for (int i = 0; i < lhs.length && comparison == 0; i++) {
  376.             append(lhs[i], rhs[i]);
  377.         }
  378.         return this;
  379.     }

  380.     /**
  381.      * Appends to the {@code builder} the comparison of
  382.      * two {@code byte}s.
  383.      *
  384.      * @param lhs  left-hand side value
  385.      * @param rhs  right-hand side value
  386.      * @return {@code this} instance.
  387.      */
  388.     public CompareToBuilder append(final byte lhs, final byte rhs) {
  389.         if (comparison != 0) {
  390.             return this;
  391.         }
  392.         comparison = Byte.compare(lhs, rhs);
  393.         return this;
  394.     }

  395.     /**
  396.      * Appends to the {@code builder} the deep comparison of
  397.      * two {@code byte} arrays.
  398.      *
  399.      * <ol>
  400.      *  <li>Check if arrays are the same using {@code ==}</li>
  401.      *  <li>Check if for {@code null}, {@code null} is less than non-{@code null}</li>
  402.      *  <li>Check array length, a shorter length array is less than a longer length array</li>
  403.      *  <li>Check array contents element by element using {@link #append(byte, byte)}</li>
  404.      * </ol>
  405.      *
  406.      * @param lhs  left-hand side array
  407.      * @param rhs  right-hand side array
  408.      * @return {@code this} instance.
  409.      */
  410.     public CompareToBuilder append(final byte[] lhs, final byte[] rhs) {
  411.         if (comparison != 0) {
  412.             return this;
  413.         }
  414.         if (lhs == rhs) {
  415.             return this;
  416.         }
  417.         if (lhs == null) {
  418.             comparison = -1;
  419.             return this;
  420.         }
  421.         if (rhs == null) {
  422.             comparison = 1;
  423.             return this;
  424.         }
  425.         if (lhs.length != rhs.length) {
  426.             comparison = lhs.length < rhs.length ? -1 : 1;
  427.             return this;
  428.         }
  429.         for (int i = 0; i < lhs.length && comparison == 0; i++) {
  430.             append(lhs[i], rhs[i]);
  431.         }
  432.         return this;
  433.     }

  434.     /**
  435.      * Appends to the {@code builder} the comparison of
  436.      * two {@code char}s.
  437.      *
  438.      * @param lhs  left-hand side value
  439.      * @param rhs  right-hand side value
  440.      * @return {@code this} instance.
  441.      */
  442.     public CompareToBuilder append(final char lhs, final char rhs) {
  443.         if (comparison != 0) {
  444.             return this;
  445.         }
  446.         comparison = Character.compare(lhs, rhs);
  447.         return this;
  448.     }

  449.     /**
  450.      * Appends to the {@code builder} the deep comparison of
  451.      * two {@code char} arrays.
  452.      *
  453.      * <ol>
  454.      *  <li>Check if arrays are the same using {@code ==}</li>
  455.      *  <li>Check if for {@code null}, {@code null} is less than non-{@code null}</li>
  456.      *  <li>Check array length, a shorter length array is less than a longer length array</li>
  457.      *  <li>Check array contents element by element using {@link #append(char, char)}</li>
  458.      * </ol>
  459.      *
  460.      * @param lhs  left-hand side array
  461.      * @param rhs  right-hand side array
  462.      * @return {@code this} instance.
  463.      */
  464.     public CompareToBuilder append(final char[] lhs, final char[] rhs) {
  465.         if (comparison != 0) {
  466.             return this;
  467.         }
  468.         if (lhs == rhs) {
  469.             return this;
  470.         }
  471.         if (lhs == null) {
  472.             comparison = -1;
  473.             return this;
  474.         }
  475.         if (rhs == null) {
  476.             comparison = 1;
  477.             return this;
  478.         }
  479.         if (lhs.length != rhs.length) {
  480.             comparison = lhs.length < rhs.length ? -1 : 1;
  481.             return this;
  482.         }
  483.         for (int i = 0; i < lhs.length && comparison == 0; i++) {
  484.             append(lhs[i], rhs[i]);
  485.         }
  486.         return this;
  487.     }

  488.     /**
  489.      * Appends to the {@code builder} the comparison of
  490.      * two {@code double}s.
  491.      *
  492.      * <p>This handles NaNs, Infinities, and {@code -0.0}.</p>
  493.      *
  494.      * <p>It is compatible with the hash code generated by
  495.      * {@link HashCodeBuilder}.</p>
  496.      *
  497.      * @param lhs  left-hand side value
  498.      * @param rhs  right-hand side value
  499.      * @return {@code this} instance.
  500.      */
  501.     public CompareToBuilder append(final double lhs, final double rhs) {
  502.         if (comparison != 0) {
  503.             return this;
  504.         }
  505.         comparison = Double.compare(lhs, rhs);
  506.         return this;
  507.     }

  508.     /**
  509.      * Appends to the {@code builder} the deep comparison of
  510.      * two {@code double} arrays.
  511.      *
  512.      * <ol>
  513.      *  <li>Check if arrays are the same using {@code ==}</li>
  514.      *  <li>Check if for {@code null}, {@code null} is less than non-{@code null}</li>
  515.      *  <li>Check array length, a shorter length array is less than a longer length array</li>
  516.      *  <li>Check array contents element by element using {@link #append(double, double)}</li>
  517.      * </ol>
  518.      *
  519.      * @param lhs  left-hand side array
  520.      * @param rhs  right-hand side array
  521.      * @return {@code this} instance.
  522.      */
  523.     public CompareToBuilder append(final double[] lhs, final double[] rhs) {
  524.         if (comparison != 0) {
  525.             return this;
  526.         }
  527.         if (lhs == rhs) {
  528.             return this;
  529.         }
  530.         if (lhs == null) {
  531.             comparison = -1;
  532.             return this;
  533.         }
  534.         if (rhs == null) {
  535.             comparison = 1;
  536.             return this;
  537.         }
  538.         if (lhs.length != rhs.length) {
  539.             comparison = lhs.length < rhs.length ? -1 : 1;
  540.             return this;
  541.         }
  542.         for (int i = 0; i < lhs.length && comparison == 0; i++) {
  543.             append(lhs[i], rhs[i]);
  544.         }
  545.         return this;
  546.     }

  547.     /**
  548.      * Appends to the {@code builder} the comparison of
  549.      * two {@code float}s.
  550.      *
  551.      * <p>This handles NaNs, Infinities, and {@code -0.0}.</p>
  552.      *
  553.      * <p>It is compatible with the hash code generated by
  554.      * {@link HashCodeBuilder}.</p>
  555.      *
  556.      * @param lhs  left-hand side value
  557.      * @param rhs  right-hand side value
  558.      * @return {@code this} instance.
  559.      */
  560.     public CompareToBuilder append(final float lhs, final float rhs) {
  561.         if (comparison != 0) {
  562.             return this;
  563.         }
  564.         comparison = Float.compare(lhs, rhs);
  565.         return this;
  566.     }

  567.     /**
  568.      * Appends to the {@code builder} the deep comparison of
  569.      * two {@code float} arrays.
  570.      *
  571.      * <ol>
  572.      *  <li>Check if arrays are the same using {@code ==}</li>
  573.      *  <li>Check if for {@code null}, {@code null} is less than non-{@code null}</li>
  574.      *  <li>Check array length, a shorter length array is less than a longer length array</li>
  575.      *  <li>Check array contents element by element using {@link #append(float, float)}</li>
  576.      * </ol>
  577.      *
  578.      * @param lhs  left-hand side array
  579.      * @param rhs  right-hand side array
  580.      * @return {@code this} instance.
  581.      */
  582.     public CompareToBuilder append(final float[] lhs, final float[] rhs) {
  583.         if (comparison != 0) {
  584.             return this;
  585.         }
  586.         if (lhs == rhs) {
  587.             return this;
  588.         }
  589.         if (lhs == null) {
  590.             comparison = -1;
  591.             return this;
  592.         }
  593.         if (rhs == null) {
  594.             comparison = 1;
  595.             return this;
  596.         }
  597.         if (lhs.length != rhs.length) {
  598.             comparison = lhs.length < rhs.length ? -1 : 1;
  599.             return this;
  600.         }
  601.         for (int i = 0; i < lhs.length && comparison == 0; i++) {
  602.             append(lhs[i], rhs[i]);
  603.         }
  604.         return this;
  605.     }

  606.     /**
  607.      * Appends to the {@code builder} the comparison of
  608.      * two {@code int}s.
  609.      *
  610.      * @param lhs  left-hand side value
  611.      * @param rhs  right-hand side value
  612.      * @return {@code this} instance.
  613.      */
  614.     public CompareToBuilder append(final int lhs, final int rhs) {
  615.         if (comparison != 0) {
  616.             return this;
  617.         }
  618.         comparison = Integer.compare(lhs, rhs);
  619.         return this;
  620.     }

  621.     /**
  622.      * Appends to the {@code builder} the deep comparison of
  623.      * two {@code int} arrays.
  624.      *
  625.      * <ol>
  626.      *  <li>Check if arrays are the same using {@code ==}</li>
  627.      *  <li>Check if for {@code null}, {@code null} is less than non-{@code null}</li>
  628.      *  <li>Check array length, a shorter length array is less than a longer length array</li>
  629.      *  <li>Check array contents element by element using {@link #append(int, int)}</li>
  630.      * </ol>
  631.      *
  632.      * @param lhs  left-hand side array
  633.      * @param rhs  right-hand side array
  634.      * @return {@code this} instance.
  635.      */
  636.     public CompareToBuilder append(final int[] lhs, final int[] rhs) {
  637.         if (comparison != 0) {
  638.             return this;
  639.         }
  640.         if (lhs == rhs) {
  641.             return this;
  642.         }
  643.         if (lhs == null) {
  644.             comparison = -1;
  645.             return this;
  646.         }
  647.         if (rhs == null) {
  648.             comparison = 1;
  649.             return this;
  650.         }
  651.         if (lhs.length != rhs.length) {
  652.             comparison = lhs.length < rhs.length ? -1 : 1;
  653.             return this;
  654.         }
  655.         for (int i = 0; i < lhs.length && comparison == 0; i++) {
  656.             append(lhs[i], rhs[i]);
  657.         }
  658.         return this;
  659.     }

  660.     /**
  661.      * Appends to the {@code builder} the comparison of
  662.      * two {@code long}s.
  663.      *
  664.      * @param lhs  left-hand side value
  665.      * @param rhs  right-hand side value
  666.      * @return {@code this} instance.
  667.      */
  668.     public CompareToBuilder append(final long lhs, final long rhs) {
  669.         if (comparison != 0) {
  670.             return this;
  671.         }
  672.         comparison = Long.compare(lhs, rhs);
  673.         return this;
  674.     }

  675.     /**
  676.      * Appends to the {@code builder} the deep comparison of
  677.      * two {@code long} arrays.
  678.      *
  679.      * <ol>
  680.      *  <li>Check if arrays are the same using {@code ==}</li>
  681.      *  <li>Check if for {@code null}, {@code null} is less than non-{@code null}</li>
  682.      *  <li>Check array length, a shorter length array is less than a longer length array</li>
  683.      *  <li>Check array contents element by element using {@link #append(long, long)}</li>
  684.      * </ol>
  685.      *
  686.      * @param lhs  left-hand side array
  687.      * @param rhs  right-hand side array
  688.      * @return {@code this} instance.
  689.      */
  690.     public CompareToBuilder append(final long[] lhs, final long[] rhs) {
  691.         if (comparison != 0) {
  692.             return this;
  693.         }
  694.         if (lhs == rhs) {
  695.             return this;
  696.         }
  697.         if (lhs == null) {
  698.             comparison = -1;
  699.             return this;
  700.         }
  701.         if (rhs == null) {
  702.             comparison = 1;
  703.             return this;
  704.         }
  705.         if (lhs.length != rhs.length) {
  706.             comparison = lhs.length < rhs.length ? -1 : 1;
  707.             return this;
  708.         }
  709.         for (int i = 0; i < lhs.length && comparison == 0; i++) {
  710.             append(lhs[i], rhs[i]);
  711.         }
  712.         return this;
  713.     }

  714.     /**
  715.      * Appends to the {@code builder} the comparison of
  716.      * two {@link Object}s.
  717.      *
  718.      * <ol>
  719.      * <li>Check if {@code lhs == rhs}</li>
  720.      * <li>Check if either {@code lhs} or {@code rhs} is {@code null},
  721.      *     a {@code null} object is less than a non-{@code null} object</li>
  722.      * <li>Check the object contents</li>
  723.      * </ol>
  724.      *
  725.      * <p>{@code lhs} must either be an array or implement {@link Comparable}.</p>
  726.      *
  727.      * @param lhs  left-hand side object
  728.      * @param rhs  right-hand side object
  729.      * @return {@code this} instance.
  730.      * @throws ClassCastException  if {@code rhs} is not assignment-compatible
  731.      *  with {@code lhs}
  732.      */
  733.     public CompareToBuilder append(final Object lhs, final Object rhs) {
  734.         return append(lhs, rhs, null);
  735.     }

  736.     /**
  737.      * Appends to the {@code builder} the comparison of
  738.      * two {@link Object}s.
  739.      *
  740.      * <ol>
  741.      * <li>Check if {@code lhs == rhs}</li>
  742.      * <li>Check if either {@code lhs} or {@code rhs} is {@code null},
  743.      *     a {@code null} object is less than a non-{@code null} object</li>
  744.      * <li>Check the object contents</li>
  745.      * </ol>
  746.      *
  747.      * <p>If {@code lhs} is an array, array comparison methods will be used.
  748.      * Otherwise {@code comparator} will be used to compare the objects.
  749.      * If {@code comparator} is {@code null}, {@code lhs} must
  750.      * implement {@link Comparable} instead.</p>
  751.      *
  752.      * @param lhs  left-hand side object
  753.      * @param rhs  right-hand side object
  754.      * @param comparator  {@link Comparator} used to compare the objects,
  755.      *  {@code null} means treat lhs as {@link Comparable}
  756.      * @return {@code this} instance.
  757.      * @throws ClassCastException  if {@code rhs} is not assignment-compatible
  758.      *  with {@code lhs}
  759.      * @since 2.0
  760.      */
  761.     public CompareToBuilder append(final Object lhs, final Object rhs, final Comparator<?> comparator) {
  762.         if (comparison != 0) {
  763.             return this;
  764.         }
  765.         if (lhs == rhs) {
  766.             return this;
  767.         }
  768.         if (lhs == null) {
  769.             comparison = -1;
  770.             return this;
  771.         }
  772.         if (rhs == null) {
  773.             comparison = 1;
  774.             return this;
  775.         }
  776.         if (ObjectUtils.isArray(lhs)) {
  777.             // factor out array case in order to keep method small enough to be inlined
  778.             appendArray(lhs, rhs, comparator);
  779.         } else // the simple case, not an array, just test the element
  780.         if (comparator == null) {
  781.             @SuppressWarnings("unchecked") // assume this can be done; if not throw CCE as per Javadoc
  782.             final Comparable<Object> comparable = (Comparable<Object>) lhs;
  783.             comparison = comparable.compareTo(rhs);
  784.         } else {
  785.             @SuppressWarnings("unchecked") // assume this can be done; if not throw CCE as per Javadoc
  786.             final Comparator<Object> comparator2 = (Comparator<Object>) comparator;
  787.             comparison = comparator2.compare(lhs, rhs);
  788.         }
  789.         return this;
  790.     }

  791.     /**
  792.      * Appends to the {@code builder} the deep comparison of
  793.      * two {@link Object} arrays.
  794.      *
  795.      * <ol>
  796.      *  <li>Check if arrays are the same using {@code ==}</li>
  797.      *  <li>Check if for {@code null}, {@code null} is less than non-{@code null}</li>
  798.      *  <li>Check array length, a short length array is less than a long length array</li>
  799.      *  <li>Check array contents element by element using {@link #append(Object, Object, Comparator)}</li>
  800.      * </ol>
  801.      *
  802.      * <p>This method will also will be called for the top level of multi-dimensional,
  803.      * ragged, and multi-typed arrays.</p>
  804.      *
  805.      * @param lhs  left-hand side array
  806.      * @param rhs  right-hand side array
  807.      * @return {@code this} instance.
  808.      * @throws ClassCastException  if {@code rhs} is not assignment-compatible
  809.      *  with {@code lhs}
  810.      */
  811.     public CompareToBuilder append(final Object[] lhs, final Object[] rhs) {
  812.         return append(lhs, rhs, null);
  813.     }

  814.     /**
  815.      * Appends to the {@code builder} the deep comparison of
  816.      * two {@link Object} arrays.
  817.      *
  818.      * <ol>
  819.      *  <li>Check if arrays are the same using {@code ==}</li>
  820.      *  <li>Check if for {@code null}, {@code null} is less than non-{@code null}</li>
  821.      *  <li>Check array length, a short length array is less than a long length array</li>
  822.      *  <li>Check array contents element by element using {@link #append(Object, Object, Comparator)}</li>
  823.      * </ol>
  824.      *
  825.      * <p>This method will also will be called for the top level of multi-dimensional,
  826.      * ragged, and multi-typed arrays.</p>
  827.      *
  828.      * @param lhs  left-hand side array
  829.      * @param rhs  right-hand side array
  830.      * @param comparator  {@link Comparator} to use to compare the array elements,
  831.      *  {@code null} means to treat {@code lhs} elements as {@link Comparable}.
  832.      * @return {@code this} instance.
  833.      * @throws ClassCastException  if {@code rhs} is not assignment-compatible
  834.      *  with {@code lhs}
  835.      * @since 2.0
  836.      */
  837.     public CompareToBuilder append(final Object[] lhs, final Object[] rhs, final Comparator<?> comparator) {
  838.         if (comparison != 0) {
  839.             return this;
  840.         }
  841.         if (lhs == rhs) {
  842.             return this;
  843.         }
  844.         if (lhs == null) {
  845.             comparison = -1;
  846.             return this;
  847.         }
  848.         if (rhs == null) {
  849.             comparison = 1;
  850.             return this;
  851.         }
  852.         if (lhs.length != rhs.length) {
  853.             comparison = lhs.length < rhs.length ? -1 : 1;
  854.             return this;
  855.         }
  856.         for (int i = 0; i < lhs.length && comparison == 0; i++) {
  857.             append(lhs[i], rhs[i], comparator);
  858.         }
  859.         return this;
  860.     }

  861.     /**
  862.      * Appends to the {@code builder} the comparison of
  863.      * two {@code short}s.
  864.      *
  865.      * @param lhs  left-hand side value
  866.      * @param rhs  right-hand side value
  867.      * @return {@code this} instance.
  868.      */
  869.     public CompareToBuilder append(final short lhs, final short rhs) {
  870.         if (comparison != 0) {
  871.             return this;
  872.         }
  873.         comparison = Short.compare(lhs, rhs);
  874.         return this;
  875.     }

  876.     /**
  877.      * Appends to the {@code builder} the deep comparison of
  878.      * two {@code short} arrays.
  879.      *
  880.      * <ol>
  881.      *  <li>Check if arrays are the same using {@code ==}</li>
  882.      *  <li>Check if for {@code null}, {@code null} is less than non-{@code null}</li>
  883.      *  <li>Check array length, a shorter length array is less than a longer length array</li>
  884.      *  <li>Check array contents element by element using {@link #append(short, short)}</li>
  885.      * </ol>
  886.      *
  887.      * @param lhs  left-hand side array
  888.      * @param rhs  right-hand side array
  889.      * @return {@code this} instance.
  890.      */
  891.     public CompareToBuilder append(final short[] lhs, final short[] rhs) {
  892.         if (comparison != 0) {
  893.             return this;
  894.         }
  895.         if (lhs == rhs) {
  896.             return this;
  897.         }
  898.         if (lhs == null) {
  899.             comparison = -1;
  900.             return this;
  901.         }
  902.         if (rhs == null) {
  903.             comparison = 1;
  904.             return this;
  905.         }
  906.         if (lhs.length != rhs.length) {
  907.             comparison = lhs.length < rhs.length ? -1 : 1;
  908.             return this;
  909.         }
  910.         for (int i = 0; i < lhs.length && comparison == 0; i++) {
  911.             append(lhs[i], rhs[i]);
  912.         }
  913.         return this;
  914.     }

  915.     private void appendArray(final Object lhs, final Object rhs, final Comparator<?> comparator) {
  916.         // switch on type of array, to dispatch to the correct handler
  917.         // handles multidimensional arrays
  918.         // throws a ClassCastException if rhs is not the correct array type
  919.         if (lhs instanceof long[]) {
  920.             append((long[]) lhs, (long[]) rhs);
  921.         } else if (lhs instanceof int[]) {
  922.             append((int[]) lhs, (int[]) rhs);
  923.         } else if (lhs instanceof short[]) {
  924.             append((short[]) lhs, (short[]) rhs);
  925.         } else if (lhs instanceof char[]) {
  926.             append((char[]) lhs, (char[]) rhs);
  927.         } else if (lhs instanceof byte[]) {
  928.             append((byte[]) lhs, (byte[]) rhs);
  929.         } else if (lhs instanceof double[]) {
  930.             append((double[]) lhs, (double[]) rhs);
  931.         } else if (lhs instanceof float[]) {
  932.             append((float[]) lhs, (float[]) rhs);
  933.         } else if (lhs instanceof boolean[]) {
  934.             append((boolean[]) lhs, (boolean[]) rhs);
  935.         } else {
  936.             // not an array of primitives
  937.             // throws a ClassCastException if rhs is not an array
  938.             append((Object[]) lhs, (Object[]) rhs, comparator);
  939.         }
  940.     }

  941.     /**
  942.      * Appends to the {@code builder} the {@code compareTo(Object)}
  943.      * result of the superclass.
  944.      *
  945.      * @param superCompareTo  result of calling {@code super.compareTo(Object)}
  946.      * @return {@code this} instance.
  947.      * @since 2.0
  948.      */
  949.     public CompareToBuilder appendSuper(final int superCompareTo) {
  950.         if (comparison != 0) {
  951.             return this;
  952.         }
  953.         comparison = superCompareTo;
  954.         return this;
  955.     }

  956.     /**
  957.      * Returns a negative Integer, a positive Integer, or zero as
  958.      * the {@code builder} has judged the "left-hand" side
  959.      * as less than, greater than, or equal to the "right-hand"
  960.      * side.
  961.      *
  962.      * @return final comparison result as an Integer
  963.      * @see #toComparison()
  964.      * @since 3.0
  965.      */
  966.     @Override
  967.     public Integer build() {
  968.         return Integer.valueOf(toComparison());
  969.     }

  970.     /**
  971.      * Returns a negative integer, a positive integer, or zero as
  972.      * the {@code builder} has judged the "left-hand" side
  973.      * as less than, greater than, or equal to the "right-hand"
  974.      * side.
  975.      *
  976.      * @return final comparison result
  977.      * @see #build()
  978.      */
  979.     public int toComparison() {
  980.         return comparison;
  981.     }
  982. }