DiffBuilder.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.util.ArrayList;
  19. import java.util.Arrays;
  20. import java.util.List;
  21. import java.util.Objects;
  22. import java.util.function.Supplier;

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

  25. /**
  26.  * Assists in implementing {@link Diffable#diff(Object)} methods.
  27.  *
  28.  * <p>
  29.  * To use this class, write code as follows:
  30.  * </p>
  31.  *
  32.  * <pre>{@code
  33.  * public class Person implements Diffable<Person> {
  34.  *   String name;
  35.  *   int age;
  36.  *   boolean smoker;
  37.  *
  38.  *   ...
  39.  *
  40.  *   public DiffResult diff(Person obj) {
  41.  *     // No need for null check, as NullPointerException correct if obj is null
  42.  *     return new DiffBuilder.<Person>builder()
  43.  *         .setLeft(this)
  44.  *         .setRight(obj)
  45.  *         .setStyle(ToStringStyle.SHORT_PREFIX_STYLE))
  46.  *         .build()
  47.  *       .append("name", this.name, obj.name)
  48.  *       .append("age", this.age, obj.age)
  49.  *       .append("smoker", this.smoker, obj.smoker)
  50.  *       .build();
  51.  *   }
  52.  * }
  53.  * }</pre>
  54.  *
  55.  * <p>
  56.  * The {@link ToStringStyle} passed to the constructor is embedded in the returned {@link DiffResult} and influences the style of the
  57.  * {@code DiffResult.toString()} method. This style choice can be overridden by calling {@link DiffResult#toString(ToStringStyle)}.
  58.  * </p>
  59.  * <p>
  60.  * See {@link ReflectionDiffBuilder} for a reflection based version of this class.
  61.  * </p>
  62.  *
  63.  * @param <T> type of the left and right object.
  64.  * @see Diffable
  65.  * @see Diff
  66.  * @see DiffResult
  67.  * @see ToStringStyle
  68.  * @see ReflectionDiffBuilder
  69.  * @since 3.3
  70.  */
  71. public class DiffBuilder<T> implements Builder<DiffResult<T>> {

  72.     /**
  73.      * Constructs a new instance.
  74.      *
  75.      * @param <T> type of the left and right object.
  76.      * @since 3.15.0
  77.      */
  78.     public static final class Builder<T> {

  79.         private T left;
  80.         private T right;
  81.         private ToStringStyle style;
  82.         private boolean testObjectsEquals = true;
  83.         private String toStringFormat = TO_STRING_FORMAT;

  84.         /**
  85.          * Constructs a new instance.
  86.          */
  87.         public Builder() {
  88.             // empty
  89.         }

  90.         /**
  91.          * Builds a new configured {@link DiffBuilder}.
  92.          *
  93.          * @return a new configured {@link DiffBuilder}.
  94.          */
  95.         public DiffBuilder<T> build() {
  96.             return new DiffBuilder<>(left, right, style, testObjectsEquals, toStringFormat);
  97.         }

  98.         /**
  99.          * Sets the left object.
  100.          *
  101.          * @param left the left object.
  102.          * @return {@code this} instance.
  103.          */
  104.         public Builder<T> setLeft(final T left) {
  105.             this.left = left;
  106.             return this;
  107.         }

  108.         /**
  109.          * Sets the right object.
  110.          *
  111.          * @param right the left object.
  112.          * @return {@code this} instance.
  113.          */
  114.         public Builder<T> setRight(final T right) {
  115.             this.right = right;
  116.             return this;
  117.         }

  118.         /**
  119.          * Sets the style will to use when outputting the objects, {@code null} uses the default.
  120.          *
  121.          * @param style the style to use when outputting the objects, {@code null} uses the default.
  122.          * @return {@code this} instance.
  123.          */
  124.         public Builder<T> setStyle(final ToStringStyle style) {
  125.             this.style = style != null ? style : ToStringStyle.DEFAULT_STYLE;
  126.             return this;
  127.         }

  128.         /**
  129.          * Sets whether to test if left and right are the same or equal. All of the append(fieldName, left, right) methods will abort without creating a field
  130.          * {@link Diff} if the trivially equal test is enabled and returns true. The result of this test is never changed throughout the life of this
  131.          * {@link DiffBuilder}.
  132.          *
  133.          * @param testObjectsEquals If true, this will test if lhs and rhs are the same or equal. All of the append(fieldName, left, right) methods will abort
  134.          *                          without creating a field {@link Diff} if the trivially equal test is enabled and returns true. The result of this test is
  135.          *                          never changed throughout the life of this {@link DiffBuilder}.
  136.          * @return {@code this} instance.
  137.          */
  138.         public Builder<T> setTestObjectsEquals(final boolean testObjectsEquals) {
  139.             this.testObjectsEquals = testObjectsEquals;
  140.             return this;
  141.         }

  142.         /**
  143.          * Sets the two-argument format string for {@link String#format(String, Object...)}, for example {@code "%s differs from %s"}.
  144.          *
  145.          * @param toStringFormat {@code null} uses the default.
  146.          * @return {@code this} instance.
  147.          */
  148.         public Builder<T> setToStringFormat(final String toStringFormat) {
  149.             this.toStringFormat = toStringFormat != null ? toStringFormat : TO_STRING_FORMAT;
  150.             return this;
  151.         }
  152.     }

  153.     private static final class SDiff<T> extends Diff<T> {

  154.         private static final long serialVersionUID = 1L;
  155.         private final transient Supplier<T> leftSupplier;
  156.         private final transient Supplier<T> rightSupplier;

  157.         private SDiff(final String fieldName, final Supplier<T> leftSupplier, final Supplier<T> rightSupplier, final Class<T> type) {
  158.             super(fieldName, type);
  159.             this.leftSupplier = Objects.requireNonNull(leftSupplier);
  160.             this.rightSupplier = Objects.requireNonNull(rightSupplier);
  161.         }

  162.         @Override
  163.         public T getLeft() {
  164.             return leftSupplier.get();
  165.         }

  166.         @Override
  167.         public T getRight() {
  168.             return rightSupplier.get();
  169.         }

  170.     }

  171.     static final String TO_STRING_FORMAT = "%s differs from %s";

  172.     /**
  173.      * Constructs a new {@link Builder}.
  174.      *
  175.      * @param <T> type of the left and right object.
  176.      * @return a new {@link Builder}.
  177.      * @since 3.15.0
  178.      */
  179.     public static <T> Builder<T> builder() {
  180.         return new Builder<>();
  181.     }

  182.     private final List<Diff<?>> diffs;
  183.     private final boolean equals;
  184.     private final T left;
  185.     private final T right;
  186.     private final ToStringStyle style;
  187.     private final String toStringFormat;

  188.     /**
  189.      * Constructs a builder for the specified objects with the specified style.
  190.      *
  191.      * <p>
  192.      * If {@code lhs == rhs} or {@code lhs.equals(rhs)} then the builder will not evaluate any calls to {@code append(...)} and will return an empty
  193.      * {@link DiffResult} when {@link #build()} is executed.
  194.      * </p>
  195.      *
  196.      * <p>
  197.      * This delegates to {@link #DiffBuilder(Object, Object, ToStringStyle, boolean)} with the testTriviallyEqual flag enabled.
  198.      * </p>
  199.      *
  200.      * @param left  {@code this} object
  201.      * @param right the object to diff against
  202.      * @param style the style to use when outputting the objects, {@code null} uses the default
  203.      * @throws NullPointerException if {@code lhs} or {@code rhs} is {@code null}
  204.      * @deprecated Use {@link Builder}.
  205.      */
  206.     @Deprecated
  207.     public DiffBuilder(final T left, final T right, final ToStringStyle style) {
  208.         this(left, right, style, true);
  209.     }

  210.     /**
  211.      * Constructs a builder for the specified objects with the specified style.
  212.      *
  213.      * <p>
  214.      * If {@code lhs == rhs} or {@code lhs.equals(rhs)} then the builder will not evaluate any calls to {@code append(...)} and will return an empty
  215.      * {@link DiffResult} when {@link #build()} is executed.
  216.      * </p>
  217.      *
  218.      * @param left              {@code this} object
  219.      * @param right             the object to diff against
  220.      * @param style             the style to use when outputting the objects, {@code null} uses the default
  221.      * @param testObjectsEquals If true, this will test if lhs and rhs are the same or equal. All of the append(fieldName, lhs, rhs) methods will abort without
  222.      *                          creating a field {@link Diff} if the trivially equal test is enabled and returns true. The result of this test is never changed
  223.      *                          throughout the life of this {@link DiffBuilder}.
  224.      * @throws NullPointerException if {@code lhs} or {@code rhs} is {@code null}
  225.      * @since 3.4
  226.      * @deprecated Use {@link Builder}.
  227.      */
  228.     @Deprecated
  229.     public DiffBuilder(final T left, final T right, final ToStringStyle style, final boolean testObjectsEquals) {
  230.         this(left, right, style, testObjectsEquals, TO_STRING_FORMAT);
  231.     }

  232.     private DiffBuilder(final T left, final T right, final ToStringStyle style, final boolean testObjectsEquals, final String toStringFormat) {
  233.         this.left = Objects.requireNonNull(left, "left");
  234.         this.right = Objects.requireNonNull(right, "right");
  235.         this.diffs = new ArrayList<>();
  236.         this.toStringFormat = toStringFormat;
  237.         this.style = style != null ? style : ToStringStyle.DEFAULT_STYLE;
  238.         // Don't compare any fields if objects equal
  239.         this.equals = testObjectsEquals && Objects.equals(left, right);
  240.     }

  241.     private <F> DiffBuilder<T> add(final String fieldName, final Supplier<F> left, final Supplier<F> right, final Class<F> type) {
  242.         diffs.add(new SDiff<>(fieldName, left, right, type));
  243.         return this;
  244.     }

  245.     /**
  246.      * Tests if two {@code boolean}s are equal.
  247.      *
  248.      * @param fieldName the field name
  249.      * @param lhs       the left-hand side {@code boolean}
  250.      * @param rhs       the right-hand side {@code boolean}
  251.      * @return {@code this} instance.
  252.      * @throws NullPointerException if field name is {@code null}
  253.      */
  254.     public DiffBuilder<T> append(final String fieldName, final boolean lhs, final boolean rhs) {
  255.         return equals || lhs == rhs ? this : add(fieldName, () -> Boolean.valueOf(lhs), () -> Boolean.valueOf(rhs), Boolean.class);
  256.     }

  257.     /**
  258.      * Tests if two {@code boolean[]}s are equal.
  259.      *
  260.      * @param fieldName the field name
  261.      * @param lhs       the left-hand side {@code boolean[]}
  262.      * @param rhs       the right-hand side {@code boolean[]}
  263.      * @return {@code this} instance.
  264.      * @throws NullPointerException if field name is {@code null}
  265.      */
  266.     public DiffBuilder<T> append(final String fieldName, final boolean[] lhs, final boolean[] rhs) {
  267.         return equals || Arrays.equals(lhs, rhs) ? this : add(fieldName, () -> ArrayUtils.toObject(lhs), () -> ArrayUtils.toObject(rhs), Boolean[].class);
  268.     }

  269.     /**
  270.      * Tests if two {@code byte}s are equal.
  271.      *
  272.      * @param fieldName the field name
  273.      * @param lhs       the left-hand side {@code byte}
  274.      * @param rhs       the right-hand side {@code byte}
  275.      * @return {@code this} instance.
  276.      * @throws NullPointerException if field name is {@code null}
  277.      */
  278.     public DiffBuilder<T> append(final String fieldName, final byte lhs, final byte rhs) {
  279.         return equals || lhs == rhs ? this : add(fieldName, () -> Byte.valueOf(lhs), () -> Byte.valueOf(rhs), Byte.class);
  280.     }

  281.     /**
  282.      * Tests if two {@code byte[]}s are equal.
  283.      *
  284.      * @param fieldName the field name
  285.      * @param lhs       the left-hand side {@code byte[]}
  286.      * @param rhs       the right-hand side {@code byte[]}
  287.      * @return {@code this} instance.
  288.      * @throws NullPointerException if field name is {@code null}
  289.      */
  290.     public DiffBuilder<T> append(final String fieldName, final byte[] lhs, final byte[] rhs) {
  291.         return equals || Arrays.equals(lhs, rhs) ? this : add(fieldName, () -> ArrayUtils.toObject(lhs), () -> ArrayUtils.toObject(rhs), Byte[].class);
  292.     }

  293.     /**
  294.      * Tests if two {@code char}s are equal.
  295.      *
  296.      * @param fieldName the field name
  297.      * @param lhs       the left-hand side {@code char}
  298.      * @param rhs       the right-hand side {@code char}
  299.      * @return {@code this} instance.
  300.      * @throws NullPointerException if field name is {@code null}
  301.      */
  302.     public DiffBuilder<T> append(final String fieldName, final char lhs, final char rhs) {
  303.         return equals || lhs == rhs ? this : add(fieldName, () -> Character.valueOf(lhs), () -> Character.valueOf(rhs), Character.class);
  304.     }

  305.     /**
  306.      * Tests if two {@code char[]}s are equal.
  307.      *
  308.      * @param fieldName the field name
  309.      * @param lhs       the left-hand side {@code char[]}
  310.      * @param rhs       the right-hand side {@code char[]}
  311.      * @return {@code this} instance.
  312.      * @throws NullPointerException if field name is {@code null}
  313.      */
  314.     public DiffBuilder<T> append(final String fieldName, final char[] lhs, final char[] rhs) {
  315.         return equals || Arrays.equals(lhs, rhs) ? this : add(fieldName, () -> ArrayUtils.toObject(lhs), () -> ArrayUtils.toObject(rhs), Character[].class);
  316.     }

  317.     /**
  318.      * Appends diffs from another {@link DiffResult}.
  319.      *
  320.      * <p>
  321.      * Useful this method to compare properties which are themselves Diffable and would like to know which specific part of it is different.
  322.      * </p>
  323.      *
  324.      * <pre>{@code
  325.      * public class Person implements Diffable<Person> {
  326.      *   String name;
  327.      *   Address address; // implements Diffable<Address>
  328.      *
  329.      *   ...
  330.      *
  331.      *   public DiffResult diff(Person obj) {
  332.      *     return new DiffBuilder(this, obj, ToStringStyle.SHORT_PREFIX_STYLE)
  333.      *       .append("name", this.name, obj.name)
  334.      *       .append("address", this.address.diff(obj.address))
  335.      *       .build();
  336.      *   }
  337.      * }
  338.      * }
  339.      * </pre>
  340.      *
  341.      * @param fieldName  the field name
  342.      * @param diffResult the {@link DiffResult} to append
  343.      * @return {@code this} instance.
  344.      * @throws NullPointerException if field name is {@code null} or diffResult is {@code null}
  345.      * @since 3.5
  346.      */
  347.     public DiffBuilder<T> append(final String fieldName, final DiffResult<?> diffResult) {
  348.         Objects.requireNonNull(diffResult, "diffResult");
  349.         if (equals) {
  350.             return this;
  351.         }
  352.         diffResult.getDiffs().forEach(diff -> append(fieldName + "." + diff.getFieldName(), diff.getLeft(), diff.getRight()));
  353.         return this;
  354.     }

  355.     /**
  356.      * Tests if two {@code double}s are equal.
  357.      *
  358.      * @param fieldName the field name
  359.      * @param lhs       the left-hand side {@code double}
  360.      * @param rhs       the right-hand side {@code double}
  361.      * @return {@code this} instance.
  362.      * @throws NullPointerException if field name is {@code null}
  363.      */
  364.     public DiffBuilder<T> append(final String fieldName, final double lhs, final double rhs) {
  365.         return equals || Double.doubleToLongBits(lhs) == Double.doubleToLongBits(rhs) ? this
  366.                 : add(fieldName, () -> Double.valueOf(lhs), () -> Double.valueOf(rhs), Double.class);
  367.     }

  368.     /**
  369.      * Tests if two {@code double[]}s are equal.
  370.      *
  371.      * @param fieldName the field name
  372.      * @param lhs       the left-hand side {@code double[]}
  373.      * @param rhs       the right-hand side {@code double[]}
  374.      * @return {@code this} instance.
  375.      * @throws NullPointerException if field name is {@code null}
  376.      */
  377.     public DiffBuilder<T> append(final String fieldName, final double[] lhs, final double[] rhs) {
  378.         return equals || Arrays.equals(lhs, rhs) ? this : add(fieldName, () -> ArrayUtils.toObject(lhs), () -> ArrayUtils.toObject(rhs), Double[].class);
  379.     }

  380.     /**
  381.      * Test if two {@code float}s are equal.
  382.      *
  383.      * @param fieldName the field name
  384.      * @param lhs       the left-hand side {@code float}
  385.      * @param rhs       the right-hand side {@code float}
  386.      * @return {@code this} instance.
  387.      * @throws NullPointerException if field name is {@code null}
  388.      */
  389.     public DiffBuilder<T> append(final String fieldName, final float lhs, final float rhs) {
  390.         return equals || Float.floatToIntBits(lhs) == Float.floatToIntBits(rhs) ? this
  391.                 : add(fieldName, () -> Float.valueOf(lhs), () -> Float.valueOf(rhs), Float.class);
  392.     }

  393.     /**
  394.      * Tests if two {@code float[]}s are equal.
  395.      *
  396.      * @param fieldName the field name
  397.      * @param lhs       the left-hand side {@code float[]}
  398.      * @param rhs       the right-hand side {@code float[]}
  399.      * @return {@code this} instance.
  400.      * @throws NullPointerException if field name is {@code null}
  401.      */
  402.     public DiffBuilder<T> append(final String fieldName, final float[] lhs, final float[] rhs) {
  403.         return equals || Arrays.equals(lhs, rhs) ? this : add(fieldName, () -> ArrayUtils.toObject(lhs), () -> ArrayUtils.toObject(rhs), Float[].class);
  404.     }

  405.     /**
  406.      * Tests if two {@code int}s are equal.
  407.      *
  408.      * @param fieldName the field name
  409.      * @param lhs       the left-hand side {@code int}
  410.      * @param rhs       the right-hand side {@code int}
  411.      * @return {@code this} instance.
  412.      * @throws NullPointerException if field name is {@code null}
  413.      */
  414.     public DiffBuilder<T> append(final String fieldName, final int lhs, final int rhs) {
  415.         return equals || lhs == rhs ? this : add(fieldName, () -> Integer.valueOf(lhs), () -> Integer.valueOf(rhs), Integer.class);
  416.     }

  417.     /**
  418.      * Tests if two {@code int[]}s are equal.
  419.      *
  420.      * @param fieldName the field name
  421.      * @param lhs       the left-hand side {@code int[]}
  422.      * @param rhs       the right-hand side {@code int[]}
  423.      * @return {@code this} instance.
  424.      * @throws NullPointerException if field name is {@code null}
  425.      */
  426.     public DiffBuilder<T> append(final String fieldName, final int[] lhs, final int[] rhs) {
  427.         return equals || Arrays.equals(lhs, rhs) ? this : add(fieldName, () -> ArrayUtils.toObject(lhs), () -> ArrayUtils.toObject(rhs), Integer[].class);
  428.     }

  429.     /**
  430.      * Tests if two {@code long}s are equal.
  431.      *
  432.      * @param fieldName the field name
  433.      * @param lhs       the left-hand side {@code long}
  434.      * @param rhs       the right-hand side {@code long}
  435.      * @return {@code this} instance.
  436.      * @throws NullPointerException if field name is {@code null}
  437.      */
  438.     public DiffBuilder<T> append(final String fieldName, final long lhs, final long rhs) {
  439.         return equals || lhs == rhs ? this : add(fieldName, () -> Long.valueOf(lhs), () -> Long.valueOf(rhs), Long.class);
  440.     }

  441.     /**
  442.      * Tests if two {@code long[]}s are equal.
  443.      *
  444.      * @param fieldName the field name
  445.      * @param lhs       the left-hand side {@code long[]}
  446.      * @param rhs       the right-hand side {@code long[]}
  447.      * @return {@code this} instance.
  448.      * @throws NullPointerException if field name is {@code null}
  449.      */
  450.     public DiffBuilder<T> append(final String fieldName, final long[] lhs, final long[] rhs) {
  451.         return equals || Arrays.equals(lhs, rhs) ? this : add(fieldName, () -> ArrayUtils.toObject(lhs), () -> ArrayUtils.toObject(rhs), Long[].class);
  452.     }

  453.     /**
  454.      * Tests if two {@link Objects}s are equal.
  455.      *
  456.      * @param fieldName the field name
  457.      * @param lhs       the left-hand side {@link Object}
  458.      * @param rhs       the right-hand side {@link Object}
  459.      * @return {@code this} instance.
  460.      * @throws NullPointerException if field name is {@code null}
  461.      */
  462.     public DiffBuilder<T> append(final String fieldName, final Object lhs, final Object rhs) {
  463.         if (equals || lhs == rhs) {
  464.             return this;
  465.         }
  466.         // rhs cannot be null, as lhs != rhs
  467.         final Object test = lhs != null ? lhs : rhs;
  468.         if (ObjectUtils.isArray(test)) {
  469.             if (test instanceof boolean[]) {
  470.                 return append(fieldName, (boolean[]) lhs, (boolean[]) rhs);
  471.             }
  472.             if (test instanceof byte[]) {
  473.                 return append(fieldName, (byte[]) lhs, (byte[]) rhs);
  474.             }
  475.             if (test instanceof char[]) {
  476.                 return append(fieldName, (char[]) lhs, (char[]) rhs);
  477.             }
  478.             if (test instanceof double[]) {
  479.                 return append(fieldName, (double[]) lhs, (double[]) rhs);
  480.             }
  481.             if (test instanceof float[]) {
  482.                 return append(fieldName, (float[]) lhs, (float[]) rhs);
  483.             }
  484.             if (test instanceof int[]) {
  485.                 return append(fieldName, (int[]) lhs, (int[]) rhs);
  486.             }
  487.             if (test instanceof long[]) {
  488.                 return append(fieldName, (long[]) lhs, (long[]) rhs);
  489.             }
  490.             if (test instanceof short[]) {
  491.                 return append(fieldName, (short[]) lhs, (short[]) rhs);
  492.             }
  493.             return append(fieldName, (Object[]) lhs, (Object[]) rhs);
  494.         }
  495.         // Not array type
  496.         return Objects.equals(lhs, rhs) ? this : add(fieldName, () -> lhs, () -> rhs, Object.class);
  497.     }

  498.     /**
  499.      * Tests if two {@code Object[]}s are equal.
  500.      *
  501.      * @param fieldName the field name
  502.      * @param lhs       the left-hand side {@code Object[]}
  503.      * @param rhs       the right-hand side {@code Object[]}
  504.      * @return {@code this} instance.
  505.      * @throws NullPointerException if field name is {@code null}
  506.      */
  507.     public DiffBuilder<T> append(final String fieldName, final Object[] lhs, final Object[] rhs) {
  508.         return equals || Arrays.equals(lhs, rhs) ? this : add(fieldName, () -> lhs, () -> rhs, Object[].class);
  509.     }

  510.     /**
  511.      * Tests if two {@code short}s are equal.
  512.      *
  513.      * @param fieldName the field name
  514.      * @param lhs       the left-hand side {@code short}
  515.      * @param rhs       the right-hand side {@code short}
  516.      * @return {@code this} instance.
  517.      * @throws NullPointerException if field name is {@code null}
  518.      */
  519.     public DiffBuilder<T> append(final String fieldName, final short lhs, final short rhs) {
  520.         return equals || lhs == rhs ? this : add(fieldName, () -> Short.valueOf(lhs), () -> Short.valueOf(rhs), Short.class);
  521.     }

  522.     /**
  523.      * Tests if two {@code short[]}s are equal.
  524.      *
  525.      * @param fieldName the field name
  526.      * @param lhs       the left-hand side {@code short[]}
  527.      * @param rhs       the right-hand side {@code short[]}
  528.      * @return {@code this} instance.
  529.      * @throws NullPointerException if field name is {@code null}
  530.      */
  531.     public DiffBuilder<T> append(final String fieldName, final short[] lhs, final short[] rhs) {
  532.         return equals || Arrays.equals(lhs, rhs) ? this : add(fieldName, () -> ArrayUtils.toObject(lhs), () -> ArrayUtils.toObject(rhs), Short[].class);
  533.     }

  534.     /**
  535.      * Builds a {@link DiffResult} based on the differences appended to this builder.
  536.      *
  537.      * @return a {@link DiffResult} containing the differences between the two objects.
  538.      */
  539.     @Override
  540.     public DiffResult<T> build() {
  541.         return new DiffResult<>(left, right, diffs, style, toStringFormat);
  542.     }

  543.     /**
  544.      * Gets the left object.
  545.      *
  546.      * @return the left object.
  547.      */
  548.     T getLeft() {
  549.         return left;
  550.     }

  551.     /**
  552.      * Gets the right object.
  553.      *
  554.      * @return the right object.
  555.      */
  556.     T getRight() {
  557.         return right;
  558.     }

  559. }