ArrayUtils.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;

  18. import java.lang.reflect.Array;
  19. import java.lang.reflect.Field;
  20. import java.lang.reflect.Method;
  21. import java.lang.reflect.Type;
  22. import java.util.Arrays;
  23. import java.util.BitSet;
  24. import java.util.Comparator;
  25. import java.util.Date;
  26. import java.util.HashMap;
  27. import java.util.Map;
  28. import java.util.Objects;
  29. import java.util.Random;
  30. import java.util.concurrent.ThreadLocalRandom;
  31. import java.util.function.Function;
  32. import java.util.function.IntFunction;
  33. import java.util.function.Supplier;

  34. import org.apache.commons.lang3.builder.EqualsBuilder;
  35. import org.apache.commons.lang3.builder.HashCodeBuilder;
  36. import org.apache.commons.lang3.builder.ToStringBuilder;
  37. import org.apache.commons.lang3.builder.ToStringStyle;
  38. import org.apache.commons.lang3.math.NumberUtils;
  39. import org.apache.commons.lang3.mutable.MutableInt;
  40. import org.apache.commons.lang3.stream.Streams;

  41. /**
  42.  * Operations on arrays, primitive arrays (like {@code int[]}) and
  43.  * primitive wrapper arrays (like {@code Integer[]}).
  44.  * <p>
  45.  * This class tries to handle {@code null} input gracefully.
  46.  * An exception will not be thrown for a {@code null}
  47.  * array input. However, an Object array that contains a {@code null}
  48.  * element may throw an exception. Each method documents its behavior.
  49.  * </p>
  50.  * <p>
  51.  * #ThreadSafe#
  52.  * </p>
  53.  * @since 2.0
  54.  */
  55. public class ArrayUtils {

  56.     /**
  57.      * An empty immutable {@code boolean} array.
  58.      */
  59.     public static final boolean[] EMPTY_BOOLEAN_ARRAY = {};

  60.     /**
  61.      * An empty immutable {@link Boolean} array.
  62.      */
  63.     public static final Boolean[] EMPTY_BOOLEAN_OBJECT_ARRAY = {};

  64.     /**
  65.      * An empty immutable {@code byte} array.
  66.      */
  67.     public static final byte[] EMPTY_BYTE_ARRAY = {};

  68.     /**
  69.      * An empty immutable {@link Byte} array.
  70.      */
  71.     public static final Byte[] EMPTY_BYTE_OBJECT_ARRAY = {};

  72.     /**
  73.      * An empty immutable {@code char} array.
  74.      */
  75.     public static final char[] EMPTY_CHAR_ARRAY = {};

  76.     /**
  77.      * An empty immutable {@link Character} array.
  78.      */
  79.     public static final Character[] EMPTY_CHARACTER_OBJECT_ARRAY = {};

  80.     /**
  81.      * An empty immutable {@link Class} array.
  82.      */
  83.     public static final Class<?>[] EMPTY_CLASS_ARRAY = {};

  84.     /**
  85.      * An empty immutable {@code double} array.
  86.      */
  87.     public static final double[] EMPTY_DOUBLE_ARRAY = {};

  88.     /**
  89.      * An empty immutable {@link Double} array.
  90.      */
  91.     public static final Double[] EMPTY_DOUBLE_OBJECT_ARRAY = {};

  92.     /**
  93.      * An empty immutable {@link Field} array.
  94.      *
  95.      * @since 3.10
  96.      */
  97.     public static final Field[] EMPTY_FIELD_ARRAY = {};

  98.     /**
  99.      * An empty immutable {@code float} array.
  100.      */
  101.     public static final float[] EMPTY_FLOAT_ARRAY = {};

  102.     /**
  103.      * An empty immutable {@link Float} array.
  104.      */
  105.     public static final Float[] EMPTY_FLOAT_OBJECT_ARRAY = {};

  106.     /**
  107.      * An empty immutable {@code int} array.
  108.      */
  109.     public static final int[] EMPTY_INT_ARRAY = {};

  110.     /**
  111.      * An empty immutable {@link Integer} array.
  112.      */
  113.     public static final Integer[] EMPTY_INTEGER_OBJECT_ARRAY = {};

  114.     /**
  115.      * An empty immutable {@code long} array.
  116.      */
  117.     public static final long[] EMPTY_LONG_ARRAY = {};

  118.     /**
  119.      * An empty immutable {@link Long} array.
  120.      */
  121.     public static final Long[] EMPTY_LONG_OBJECT_ARRAY = {};

  122.     /**
  123.      * An empty immutable {@link Method} array.
  124.      *
  125.      * @since 3.10
  126.      */
  127.     public static final Method[] EMPTY_METHOD_ARRAY = {};

  128.     /**
  129.      * An empty immutable {@link Object} array.
  130.      */
  131.     public static final Object[] EMPTY_OBJECT_ARRAY = {};

  132.     /**
  133.      * An empty immutable {@code short} array.
  134.      */
  135.     public static final short[] EMPTY_SHORT_ARRAY = {};

  136.     /**
  137.      * An empty immutable {@link Short} array.
  138.      */
  139.     public static final Short[] EMPTY_SHORT_OBJECT_ARRAY = {};

  140.     /**
  141.      * An empty immutable {@link String} array.
  142.      */
  143.     public static final String[] EMPTY_STRING_ARRAY = {};

  144.     /**
  145.      * An empty immutable {@link Throwable} array.
  146.      *
  147.      * @since 3.10
  148.      */
  149.     public static final Throwable[] EMPTY_THROWABLE_ARRAY = {};

  150.     /**
  151.      * An empty immutable {@link Type} array.
  152.      *
  153.      * @since 3.10
  154.      */
  155.     public static final Type[] EMPTY_TYPE_ARRAY = {};

  156.     /**
  157.      * The index value when an element is not found in a list or array: {@code -1}.
  158.      * This value is returned by methods in this class and can also be used in comparisons with values returned by
  159.      * various method from {@link java.util.List}.
  160.      */
  161.     public static final int INDEX_NOT_FOUND = -1;

  162.     /**
  163.      * Copies the given array and adds the given element at the end of the new array.
  164.      * <p>
  165.      * The new array contains the same elements of the input
  166.      * array plus the given element in the last position. The component type of
  167.      * the new array is the same as that of the input array.
  168.      * </p>
  169.      * <p>
  170.      * If the input array is {@code null}, a new one element array is returned
  171.      * whose component type is the same as the element.
  172.      * </p>
  173.      * <pre>
  174.      * ArrayUtils.add(null, true)          = [true]
  175.      * ArrayUtils.add([true], false)       = [true, false]
  176.      * ArrayUtils.add([true, false], true) = [true, false, true]
  177.      * </pre>
  178.      *
  179.      * @param array  the array to copy and add the element to, may be {@code null}
  180.      * @param element  the object to add at the last index of the new array
  181.      * @return A new array containing the existing elements plus the new element
  182.      * @since 2.1
  183.      */
  184.     public static boolean[] add(final boolean[] array, final boolean element) {
  185.         final boolean[] newArray = (boolean[]) copyArrayGrow1(array, Boolean.TYPE);
  186.         newArray[newArray.length - 1] = element;
  187.         return newArray;
  188.     }

  189.     /**
  190.      * Inserts the specified element at the specified position in the array.
  191.      * Shifts the element currently at that position (if any) and any subsequent
  192.      * elements to the right (adds one to their indices).
  193.      * <p>
  194.      * This method returns a new array with the same elements of the input
  195.      * array plus the given element on the specified position. The component
  196.      * type of the returned array is always the same as that of the input
  197.      * array.
  198.      * </p>
  199.      * <p>
  200.      * If the input array is {@code null}, a new one element array is returned
  201.      * whose component type is the same as the element.
  202.      * </p>
  203.      * <pre>
  204.      * ArrayUtils.add(null, 0, true)          = [true]
  205.      * ArrayUtils.add([true], 0, false)       = [false, true]
  206.      * ArrayUtils.add([false], 1, true)       = [false, true]
  207.      * ArrayUtils.add([true, false], 1, true) = [true, true, false]
  208.      * </pre>
  209.      *
  210.      * @param array  the array to add the element to, may be {@code null}
  211.      * @param index  the position of the new object
  212.      * @param element  the object to add
  213.      * @return A new array containing the existing elements and the new element
  214.      * @throws IndexOutOfBoundsException if the index is out of range (index &lt; 0 || index &gt; array.length).
  215.      * @deprecated this method has been superseded by {@link #insert(int, boolean[], boolean...)} and
  216.      * may be removed in a future release. Please note the handling of {@code null} input arrays differs
  217.      * in the new method: inserting {@code X} into a {@code null} array results in {@code null} not {@code X}.
  218.      */
  219.     @Deprecated
  220.     public static boolean[] add(final boolean[] array, final int index, final boolean element) {
  221.         return (boolean[]) add(array, index, Boolean.valueOf(element), Boolean.TYPE);
  222.     }

  223.     /**
  224.      * Copies the given array and adds the given element at the end of the new array.
  225.      * <p>
  226.      * The new array contains the same elements of the input
  227.      * array plus the given element in the last position. The component type of
  228.      * the new array is the same as that of the input array.
  229.      * </p>
  230.      * <p>
  231.      * If the input array is {@code null}, a new one element array is returned
  232.      * whose component type is the same as the element.
  233.      * </p>
  234.      * <pre>
  235.      * ArrayUtils.add(null, 0)   = [0]
  236.      * ArrayUtils.add([1], 0)    = [1, 0]
  237.      * ArrayUtils.add([1, 0], 1) = [1, 0, 1]
  238.      * </pre>
  239.      *
  240.      * @param array  the array to copy and add the element to, may be {@code null}
  241.      * @param element  the object to add at the last index of the new array
  242.      * @return A new array containing the existing elements plus the new element
  243.      * @since 2.1
  244.      */
  245.     public static byte[] add(final byte[] array, final byte element) {
  246.         final byte[] newArray = (byte[]) copyArrayGrow1(array, Byte.TYPE);
  247.         newArray[newArray.length - 1] = element;
  248.         return newArray;
  249.     }

  250.     /**
  251.      * Inserts the specified element at the specified position in the array.
  252.      * Shifts the element currently at that position (if any) and any subsequent
  253.      * elements to the right (adds one to their indices).
  254.      * <p>
  255.      * This method returns a new array with the same elements of the input
  256.      * array plus the given element on the specified position. The component
  257.      * type of the returned array is always the same as that of the input
  258.      * array.
  259.      * </p>
  260.      * <p>
  261.      * If the input array is {@code null}, a new one element array is returned
  262.      * whose component type is the same as the element.
  263.      * </p>
  264.      * <pre>
  265.      * ArrayUtils.add([1], 0, 2)         = [2, 1]
  266.      * ArrayUtils.add([2, 6], 2, 3)      = [2, 6, 3]
  267.      * ArrayUtils.add([2, 6], 0, 1)      = [1, 2, 6]
  268.      * ArrayUtils.add([2, 6, 3], 2, 1)   = [2, 6, 1, 3]
  269.      * </pre>
  270.      *
  271.      * @param array  the array to add the element to, may be {@code null}
  272.      * @param index  the position of the new object
  273.      * @param element  the object to add
  274.      * @return A new array containing the existing elements and the new element
  275.      * @throws IndexOutOfBoundsException if the index is out of range
  276.      * (index &lt; 0 || index &gt; array.length).
  277.      * @deprecated this method has been superseded by {@link #insert(int, byte[], byte...)} and
  278.      * may be removed in a future release. Please note the handling of {@code null} input arrays differs
  279.      * in the new method: inserting {@code X} into a {@code null} array results in {@code null} not {@code X}.
  280.      */
  281.     @Deprecated
  282.     public static byte[] add(final byte[] array, final int index, final byte element) {
  283.         return (byte[]) add(array, index, Byte.valueOf(element), Byte.TYPE);
  284.     }

  285.     /**
  286.      * Copies the given array and adds the given element at the end of the new array.
  287.      * <p>
  288.      * The new array contains the same elements of the input
  289.      * array plus the given element in the last position. The component type of
  290.      * the new array is the same as that of the input array.
  291.      * </p>
  292.      * <p>
  293.      * If the input array is {@code null}, a new one element array is returned
  294.      * whose component type is the same as the element.
  295.      * </p>
  296.      * <pre>
  297.      * ArrayUtils.add(null, '0')       = ['0']
  298.      * ArrayUtils.add(['1'], '0')      = ['1', '0']
  299.      * ArrayUtils.add(['1', '0'], '1') = ['1', '0', '1']
  300.      * </pre>
  301.      *
  302.      * @param array  the array to copy and add the element to, may be {@code null}
  303.      * @param element  the object to add at the last index of the new array
  304.      * @return A new array containing the existing elements plus the new element
  305.      * @since 2.1
  306.      */
  307.     public static char[] add(final char[] array, final char element) {
  308.         final char[] newArray = (char[]) copyArrayGrow1(array, Character.TYPE);
  309.         newArray[newArray.length - 1] = element;
  310.         return newArray;
  311.     }

  312.     /**
  313.      * Inserts the specified element at the specified position in the array.
  314.      * Shifts the element currently at that position (if any) and any subsequent
  315.      * elements to the right (adds one to their indices).
  316.      * <p>
  317.      * This method returns a new array with the same elements of the input
  318.      * array plus the given element on the specified position. The component
  319.      * type of the returned array is always the same as that of the input
  320.      * array.
  321.      * </p>
  322.      * <p>
  323.      * If the input array is {@code null}, a new one element array is returned
  324.      * whose component type is the same as the element.
  325.      * </p>
  326.      * <pre>
  327.      * ArrayUtils.add(null, 0, 'a')            = ['a']
  328.      * ArrayUtils.add(['a'], 0, 'b')           = ['b', 'a']
  329.      * ArrayUtils.add(['a', 'b'], 0, 'c')      = ['c', 'a', 'b']
  330.      * ArrayUtils.add(['a', 'b'], 1, 'k')      = ['a', 'k', 'b']
  331.      * ArrayUtils.add(['a', 'b', 'c'], 1, 't') = ['a', 't', 'b', 'c']
  332.      * </pre>
  333.      *
  334.      * @param array  the array to add the element to, may be {@code null}
  335.      * @param index  the position of the new object
  336.      * @param element  the object to add
  337.      * @return A new array containing the existing elements and the new element
  338.      * @throws IndexOutOfBoundsException if the index is out of range
  339.      * (index &lt; 0 || index &gt; array.length).
  340.      * @deprecated this method has been superseded by {@link #insert(int, char[], char...)} and
  341.      * may be removed in a future release. Please note the handling of {@code null} input arrays differs
  342.      * in the new method: inserting {@code X} into a {@code null} array results in {@code null} not {@code X}.
  343.      */
  344.     @Deprecated
  345.     public static char[] add(final char[] array, final int index, final char element) {
  346.         return (char[]) add(array, index, Character.valueOf(element), Character.TYPE);
  347.     }

  348.     /**
  349.      * Copies the given array and adds the given element at the end of the new array.
  350.      *
  351.      * <p>
  352.      * The new array contains the same elements of the input
  353.      * array plus the given element in the last position. The component type of
  354.      * the new array is the same as that of the input array.
  355.      * </p>
  356.      * <p>
  357.      * If the input array is {@code null}, a new one element array is returned
  358.      * whose component type is the same as the element.
  359.      * </p>
  360.      * <pre>
  361.      * ArrayUtils.add(null, 0)   = [0]
  362.      * ArrayUtils.add([1], 0)    = [1, 0]
  363.      * ArrayUtils.add([1, 0], 1) = [1, 0, 1]
  364.      * </pre>
  365.      *
  366.      * @param array  the array to copy and add the element to, may be {@code null}
  367.      * @param element  the object to add at the last index of the new array
  368.      * @return A new array containing the existing elements plus the new element
  369.      * @since 2.1
  370.      */
  371.     public static double[] add(final double[] array, final double element) {
  372.         final double[] newArray = (double[]) copyArrayGrow1(array, Double.TYPE);
  373.         newArray[newArray.length - 1] = element;
  374.         return newArray;
  375.     }

  376.     /**
  377.      * Inserts the specified element at the specified position in the array.
  378.      * Shifts the element currently at that position (if any) and any subsequent
  379.      * elements to the right (adds one to their indices).
  380.      * <p>
  381.      * This method returns a new array with the same elements of the input
  382.      * array plus the given element on the specified position. The component
  383.      * type of the returned array is always the same as that of the input
  384.      * array.
  385.      * </p>
  386.      * <p>
  387.      * If the input array is {@code null}, a new one element array is returned
  388.      * whose component type is the same as the element.
  389.      * </p>
  390.      * <pre>
  391.      * ArrayUtils.add([1.1], 0, 2.2)              = [2.2, 1.1]
  392.      * ArrayUtils.add([2.3, 6.4], 2, 10.5)        = [2.3, 6.4, 10.5]
  393.      * ArrayUtils.add([2.6, 6.7], 0, -4.8)        = [-4.8, 2.6, 6.7]
  394.      * ArrayUtils.add([2.9, 6.0, 0.3], 2, 1.0)    = [2.9, 6.0, 1.0, 0.3]
  395.      * </pre>
  396.      *
  397.      * @param array  the array to add the element to, may be {@code null}
  398.      * @param index  the position of the new object
  399.      * @param element  the object to add
  400.      * @return A new array containing the existing elements and the new element
  401.      * @throws IndexOutOfBoundsException if the index is out of range
  402.      * (index &lt; 0 || index &gt; array.length).
  403.      * @deprecated this method has been superseded by {@link #insert(int, double[], double...)} and
  404.      * may be removed in a future release. Please note the handling of {@code null} input arrays differs
  405.      * in the new method: inserting {@code X} into a {@code null} array results in {@code null} not {@code X}.
  406.      */
  407.     @Deprecated
  408.     public static double[] add(final double[] array, final int index, final double element) {
  409.         return (double[]) add(array, index, Double.valueOf(element), Double.TYPE);
  410.     }

  411.     /**
  412.      * Copies the given array and adds the given element at the end of the new array.
  413.      * <p>
  414.      * The new array contains the same elements of the input
  415.      * array plus the given element in the last position. The component type of
  416.      * the new array is the same as that of the input array.
  417.      * </p>
  418.      * <p>
  419.      * If the input array is {@code null}, a new one element array is returned
  420.      * whose component type is the same as the element.
  421.      * </p>
  422.      * <pre>
  423.      * ArrayUtils.add(null, 0)   = [0]
  424.      * ArrayUtils.add([1], 0)    = [1, 0]
  425.      * ArrayUtils.add([1, 0], 1) = [1, 0, 1]
  426.      * </pre>
  427.      *
  428.      * @param array  the array to copy and add the element to, may be {@code null}
  429.      * @param element  the object to add at the last index of the new array
  430.      * @return A new array containing the existing elements plus the new element
  431.      * @since 2.1
  432.      */
  433.     public static float[] add(final float[] array, final float element) {
  434.         final float[] newArray = (float[]) copyArrayGrow1(array, Float.TYPE);
  435.         newArray[newArray.length - 1] = element;
  436.         return newArray;
  437.     }

  438.     /**
  439.      * Inserts the specified element at the specified position in the array.
  440.      * Shifts the element currently at that position (if any) and any subsequent
  441.      * elements to the right (adds one to their indices).
  442.      * <p>
  443.      * This method returns a new array with the same elements of the input
  444.      * array plus the given element on the specified position. The component
  445.      * type of the returned array is always the same as that of the input
  446.      * array.
  447.      * </p>
  448.      * <p>
  449.      * If the input array is {@code null}, a new one element array is returned
  450.      * whose component type is the same as the element.
  451.      * </p>
  452.      * <pre>
  453.      * ArrayUtils.add([1.1f], 0, 2.2f)               = [2.2f, 1.1f]
  454.      * ArrayUtils.add([2.3f, 6.4f], 2, 10.5f)        = [2.3f, 6.4f, 10.5f]
  455.      * ArrayUtils.add([2.6f, 6.7f], 0, -4.8f)        = [-4.8f, 2.6f, 6.7f]
  456.      * ArrayUtils.add([2.9f, 6.0f, 0.3f], 2, 1.0f)   = [2.9f, 6.0f, 1.0f, 0.3f]
  457.      * </pre>
  458.      *
  459.      * @param array  the array to add the element to, may be {@code null}
  460.      * @param index  the position of the new object
  461.      * @param element  the object to add
  462.      * @return A new array containing the existing elements and the new element
  463.      * @throws IndexOutOfBoundsException if the index is out of range
  464.      * (index &lt; 0 || index &gt; array.length).
  465.      * @deprecated this method has been superseded by {@link #insert(int, float[], float...)} and
  466.      * may be removed in a future release. Please note the handling of {@code null} input arrays differs
  467.      * in the new method: inserting {@code X} into a {@code null} array results in {@code null} not {@code X}.
  468.      */
  469.     @Deprecated
  470.     public static float[] add(final float[] array, final int index, final float element) {
  471.         return (float[]) add(array, index, Float.valueOf(element), Float.TYPE);
  472.     }

  473.     /**
  474.      * Copies the given array and adds the given element at the end of the new array.
  475.      * <p>
  476.      * The new array contains the same elements of the input
  477.      * array plus the given element in the last position. The component type of
  478.      * the new array is the same as that of the input array.
  479.      * </p>
  480.      * <p>
  481.      * If the input array is {@code null}, a new one element array is returned
  482.      * whose component type is the same as the element.
  483.      * </p>
  484.      * <pre>
  485.      * ArrayUtils.add(null, 0)   = [0]
  486.      * ArrayUtils.add([1], 0)    = [1, 0]
  487.      * ArrayUtils.add([1, 0], 1) = [1, 0, 1]
  488.      * </pre>
  489.      *
  490.      * @param array  the array to copy and add the element to, may be {@code null}
  491.      * @param element  the object to add at the last index of the new array
  492.      * @return A new array containing the existing elements plus the new element
  493.      * @since 2.1
  494.      */
  495.     public static int[] add(final int[] array, final int element) {
  496.         final int[] newArray = (int[]) copyArrayGrow1(array, Integer.TYPE);
  497.         newArray[newArray.length - 1] = element;
  498.         return newArray;
  499.     }

  500.     /**
  501.      * Inserts the specified element at the specified position in the array.
  502.      * Shifts the element currently at that position (if any) and any subsequent
  503.      * elements to the right (adds one to their indices).
  504.      * <p>
  505.      * This method returns a new array with the same elements of the input
  506.      * array plus the given element on the specified position. The component
  507.      * type of the returned array is always the same as that of the input
  508.      * array.
  509.      * </p>
  510.      * <p>
  511.      * If the input array is {@code null}, a new one element array is returned
  512.      * whose component type is the same as the element.
  513.      * </p>
  514.      * <pre>
  515.      * ArrayUtils.add([1], 0, 2)         = [2, 1]
  516.      * ArrayUtils.add([2, 6], 2, 10)     = [2, 6, 10]
  517.      * ArrayUtils.add([2, 6], 0, -4)     = [-4, 2, 6]
  518.      * ArrayUtils.add([2, 6, 3], 2, 1)   = [2, 6, 1, 3]
  519.      * </pre>
  520.      *
  521.      * @param array  the array to add the element to, may be {@code null}
  522.      * @param index  the position of the new object
  523.      * @param element  the object to add
  524.      * @return A new array containing the existing elements and the new element
  525.      * @throws IndexOutOfBoundsException if the index is out of range
  526.      * (index &lt; 0 || index &gt; array.length).
  527.      * @deprecated this method has been superseded by {@link #insert(int, int[], int...)} and
  528.      * may be removed in a future release. Please note the handling of {@code null} input arrays differs
  529.      * in the new method: inserting {@code X} into a {@code null} array results in {@code null} not {@code X}.
  530.      */
  531.     @Deprecated
  532.     public static int[] add(final int[] array, final int index, final int element) {
  533.         return (int[]) add(array, index, Integer.valueOf(element), Integer.TYPE);
  534.     }

  535.     /**
  536.      * Inserts the specified element at the specified position in the array.
  537.      * Shifts the element currently at that position (if any) and any subsequent
  538.      * elements to the right (adds one to their indices).
  539.      * <p>
  540.      * This method returns a new array with the same elements of the input
  541.      * array plus the given element on the specified position. The component
  542.      * type of the returned array is always the same as that of the input
  543.      * array.
  544.      * </p>
  545.      * <p>
  546.      * If the input array is {@code null}, a new one element array is returned
  547.      * whose component type is the same as the element.
  548.      * </p>
  549.      * <pre>
  550.      * ArrayUtils.add([1L], 0, 2L)           = [2L, 1L]
  551.      * ArrayUtils.add([2L, 6L], 2, 10L)      = [2L, 6L, 10L]
  552.      * ArrayUtils.add([2L, 6L], 0, -4L)      = [-4L, 2L, 6L]
  553.      * ArrayUtils.add([2L, 6L, 3L], 2, 1L)   = [2L, 6L, 1L, 3L]
  554.      * </pre>
  555.      *
  556.      * @param array  the array to add the element to, may be {@code null}
  557.      * @param index  the position of the new object
  558.      * @param element  the object to add
  559.      * @return A new array containing the existing elements and the new element
  560.      * @throws IndexOutOfBoundsException if the index is out of range
  561.      * (index &lt; 0 || index &gt; array.length).
  562.      * @deprecated this method has been superseded by {@link #insert(int, long[], long...)} and
  563.      * may be removed in a future release. Please note the handling of {@code null} input arrays differs
  564.      * in the new method: inserting {@code X} into a {@code null} array results in {@code null} not {@code X}.
  565.      */
  566.     @Deprecated
  567.     public static long[] add(final long[] array, final int index, final long element) {
  568.         return (long[]) add(array, index, Long.valueOf(element), Long.TYPE);
  569.     }

  570.     /**
  571.      * Copies the given array and adds the given element at the end of the new array.
  572.      * <p>
  573.      * The new array contains the same elements of the input
  574.      * array plus the given element in the last position. The component type of
  575.      * the new array is the same as that of the input array.
  576.      * </p>
  577.      * <p>
  578.      * If the input array is {@code null}, a new one element array is returned
  579.      * whose component type is the same as the element.
  580.      * </p>
  581.      * <pre>
  582.      * ArrayUtils.add(null, 0)   = [0]
  583.      * ArrayUtils.add([1], 0)    = [1, 0]
  584.      * ArrayUtils.add([1, 0], 1) = [1, 0, 1]
  585.      * </pre>
  586.      *
  587.      * @param array  the array to copy and add the element to, may be {@code null}
  588.      * @param element  the object to add at the last index of the new array
  589.      * @return A new array containing the existing elements plus the new element
  590.      * @since 2.1
  591.      */
  592.     public static long[] add(final long[] array, final long element) {
  593.         final long[] newArray = (long[]) copyArrayGrow1(array, Long.TYPE);
  594.         newArray[newArray.length - 1] = element;
  595.         return newArray;
  596.     }

  597.     /**
  598.      * Underlying implementation of add(array, index, element) methods.
  599.      * The last parameter is the class, which may not equal element.getClass
  600.      * for primitives.
  601.      *
  602.      * @param array  the array to add the element to, may be {@code null}
  603.      * @param index  the position of the new object
  604.      * @param element  the object to add
  605.      * @param clazz the type of the element being added
  606.      * @return A new array containing the existing elements and the new element
  607.      */
  608.     private static Object add(final Object array, final int index, final Object element, final Class<?> clazz) {
  609.         if (array == null) {
  610.             if (index != 0) {
  611.                 throw new IndexOutOfBoundsException("Index: " + index + ", Length: 0");
  612.             }
  613.             final Object joinedArray = Array.newInstance(clazz, 1);
  614.             Array.set(joinedArray, 0, element);
  615.             return joinedArray;
  616.         }
  617.         final int length = Array.getLength(array);
  618.         if (index > length || index < 0) {
  619.             throw new IndexOutOfBoundsException("Index: " + index + ", Length: " + length);
  620.         }
  621.         final Object result = arraycopy(array, 0, 0, index, () -> Array.newInstance(clazz, length + 1));
  622.         Array.set(result, index, element);
  623.         if (index < length) {
  624.             System.arraycopy(array, index, result, index + 1, length - index);
  625.         }
  626.         return result;
  627.     }

  628.     /**
  629.      * Inserts the specified element at the specified position in the array.
  630.      * Shifts the element currently at that position (if any) and any subsequent
  631.      * elements to the right (adds one to their indices).
  632.      * <p>
  633.      * This method returns a new array with the same elements of the input
  634.      * array plus the given element on the specified position. The component
  635.      * type of the returned array is always the same as that of the input
  636.      * array.
  637.      * </p>
  638.      * <p>
  639.      * If the input array is {@code null}, a new one element array is returned
  640.      * whose component type is the same as the element.
  641.      * </p>
  642.      * <pre>
  643.      * ArrayUtils.add([1], 0, 2)         = [2, 1]
  644.      * ArrayUtils.add([2, 6], 2, 10)     = [2, 6, 10]
  645.      * ArrayUtils.add([2, 6], 0, -4)     = [-4, 2, 6]
  646.      * ArrayUtils.add([2, 6, 3], 2, 1)   = [2, 6, 1, 3]
  647.      * </pre>
  648.      *
  649.      * @param array  the array to add the element to, may be {@code null}
  650.      * @param index  the position of the new object
  651.      * @param element  the object to add
  652.      * @return A new array containing the existing elements and the new element
  653.      * @throws IndexOutOfBoundsException if the index is out of range
  654.      * (index &lt; 0 || index &gt; array.length).
  655.      * @deprecated this method has been superseded by {@link #insert(int, short[], short...)} and
  656.      * may be removed in a future release. Please note the handling of {@code null} input arrays differs
  657.      * in the new method: inserting {@code X} into a {@code null} array results in {@code null} not {@code X}.
  658.      */
  659.     @Deprecated
  660.     public static short[] add(final short[] array, final int index, final short element) {
  661.         return (short[]) add(array, index, Short.valueOf(element), Short.TYPE);
  662.     }

  663.     /**
  664.      * Copies the given array and adds the given element at the end of the new array.
  665.      * <p>
  666.      * The new array contains the same elements of the input
  667.      * array plus the given element in the last position. The component type of
  668.      * the new array is the same as that of the input array.
  669.      * </p>
  670.      * <p>
  671.      * If the input array is {@code null}, a new one element array is returned
  672.      * whose component type is the same as the element.
  673.      * </p>
  674.      * <pre>
  675.      * ArrayUtils.add(null, 0)   = [0]
  676.      * ArrayUtils.add([1], 0)    = [1, 0]
  677.      * ArrayUtils.add([1, 0], 1) = [1, 0, 1]
  678.      * </pre>
  679.      *
  680.      * @param array  the array to copy and add the element to, may be {@code null}
  681.      * @param element  the object to add at the last index of the new array
  682.      * @return A new array containing the existing elements plus the new element
  683.      * @since 2.1
  684.      */
  685.     public static short[] add(final short[] array, final short element) {
  686.         final short[] newArray = (short[]) copyArrayGrow1(array, Short.TYPE);
  687.         newArray[newArray.length - 1] = element;
  688.         return newArray;
  689.     }

  690.     /**
  691.      * Inserts the specified element at the specified position in the array.
  692.      * Shifts the element currently at that position (if any) and any subsequent
  693.      * elements to the right (adds one to their indices).
  694.      * <p>
  695.      * This method returns a new array with the same elements of the input
  696.      * array plus the given element on the specified position. The component
  697.      * type of the returned array is always the same as that of the input
  698.      * array.
  699.      * </p>
  700.      * <p>
  701.      * If the input array is {@code null}, a new one element array is returned
  702.      * whose component type is the same as the element.
  703.      * </p>
  704.      * <pre>
  705.      * ArrayUtils.add(null, 0, null)      = IllegalArgumentException
  706.      * ArrayUtils.add(null, 0, "a")       = ["a"]
  707.      * ArrayUtils.add(["a"], 1, null)     = ["a", null]
  708.      * ArrayUtils.add(["a"], 1, "b")      = ["a", "b"]
  709.      * ArrayUtils.add(["a", "b"], 3, "c") = ["a", "b", "c"]
  710.      * </pre>
  711.      *
  712.      * @param <T> the component type of the array
  713.      * @param array  the array to add the element to, may be {@code null}
  714.      * @param index  the position of the new object
  715.      * @param element  the object to add
  716.      * @return A new array containing the existing elements and the new element
  717.      * @throws IndexOutOfBoundsException if the index is out of range (index &lt; 0 || index &gt; array.length).
  718.      * @throws IllegalArgumentException if both array and element are null
  719.      * @deprecated this method has been superseded by {@link #insert(int, Object[], Object...) insert(int, T[], T...)} and
  720.      * may be removed in a future release. Please note the handling of {@code null} input arrays differs
  721.      * in the new method: inserting {@code X} into a {@code null} array results in {@code null} not {@code X}.
  722.      */
  723.     @Deprecated
  724.     public static <T> T[] add(final T[] array, final int index, final T element) {
  725.         final Class<T> clazz;
  726.         if (array != null) {
  727.             clazz = getComponentType(array);
  728.         } else if (element != null) {
  729.             clazz = ObjectUtils.getClass(element);
  730.         } else {
  731.             throw new IllegalArgumentException("Array and element cannot both be null");
  732.         }
  733.         return (T[]) add(array, index, element, clazz);
  734.     }

  735.     /**
  736.      * Copies the given array and adds the given element at the end of the new array.
  737.      * <p>
  738.      * The new array contains the same elements of the input
  739.      * array plus the given element in the last position. The component type of
  740.      * the new array is the same as that of the input array.
  741.      * </p>
  742.      * <p>
  743.      * If the input array is {@code null}, a new one element array is returned
  744.      * whose component type is the same as the element, unless the element itself is null,
  745.      * in which case the return type is Object[]
  746.      * </p>
  747.      * <pre>
  748.      * ArrayUtils.add(null, null)      = IllegalArgumentException
  749.      * ArrayUtils.add(null, "a")       = ["a"]
  750.      * ArrayUtils.add(["a"], null)     = ["a", null]
  751.      * ArrayUtils.add(["a"], "b")      = ["a", "b"]
  752.      * ArrayUtils.add(["a", "b"], "c") = ["a", "b", "c"]
  753.      * </pre>
  754.      *
  755.      * @param <T> the component type of the array
  756.      * @param array  the array to "add" the element to, may be {@code null}
  757.      * @param element  the object to add, may be {@code null}
  758.      * @return A new array containing the existing elements plus the new element
  759.      * The returned array type will be that of the input array (unless null),
  760.      * in which case it will have the same type as the element.
  761.      * If both are null, an IllegalArgumentException is thrown
  762.      * @since 2.1
  763.      * @throws IllegalArgumentException if both arguments are null
  764.      */
  765.     public static <T> T[] add(final T[] array, final T element) {
  766.         final Class<?> type;
  767.         if (array != null) {
  768.             type = array.getClass().getComponentType();
  769.         } else if (element != null) {
  770.             type = element.getClass();
  771.         } else {
  772.             throw new IllegalArgumentException("Arguments cannot both be null");
  773.         }
  774.         @SuppressWarnings("unchecked") // type must be T
  775.         final
  776.         T[] newArray = (T[]) copyArrayGrow1(array, type);
  777.         newArray[newArray.length - 1] = element;
  778.         return newArray;
  779.     }

  780.     /**
  781.      * Adds all the elements of the given arrays into a new array.
  782.      * <p>
  783.      * The new array contains all of the element of {@code array1} followed
  784.      * by all of the elements {@code array2}. When an array is returned, it is always
  785.      * a new array.
  786.      * </p>
  787.      * <pre>
  788.      * ArrayUtils.addAll(array1, null)   = cloned copy of array1
  789.      * ArrayUtils.addAll(null, array2)   = cloned copy of array2
  790.      * ArrayUtils.addAll([], [])         = []
  791.      * ArrayUtils.addAll(null, null)     = null
  792.      * </pre>
  793.      *
  794.      * @param array1  the first array whose elements are added to the new array.
  795.      * @param array2  the second array whose elements are added to the new array.
  796.      * @return The new boolean[] array or {@code null}.
  797.      * @since 2.1
  798.      */
  799.     public static boolean[] addAll(final boolean[] array1, final boolean... array2) {
  800.         if (array1 == null) {
  801.             return clone(array2);
  802.         }
  803.         if (array2 == null) {
  804.             return clone(array1);
  805.         }
  806.         final boolean[] joinedArray = new boolean[array1.length + array2.length];
  807.         System.arraycopy(array1, 0, joinedArray, 0, array1.length);
  808.         System.arraycopy(array2, 0, joinedArray, array1.length, array2.length);
  809.         return joinedArray;
  810.     }

  811.     /**
  812.      * Adds all the elements of the given arrays into a new array.
  813.      * <p>
  814.      * The new array contains all of the element of {@code array1} followed
  815.      * by all of the elements {@code array2}. When an array is returned, it is always
  816.      * a new array.
  817.      * </p>
  818.      * <pre>
  819.      * ArrayUtils.addAll(array1, null)   = cloned copy of array1
  820.      * ArrayUtils.addAll(null, array2)   = cloned copy of array2
  821.      * ArrayUtils.addAll([], [])         = []
  822.      * ArrayUtils.addAll(null, null)     = null
  823.      * </pre>
  824.      *
  825.      * @param array1  the first array whose elements are added to the new array.
  826.      * @param array2  the second array whose elements are added to the new array.
  827.      * @return The new byte[] array or {@code null}.
  828.      * @since 2.1
  829.      */
  830.     public static byte[] addAll(final byte[] array1, final byte... array2) {
  831.         if (array1 == null) {
  832.             return clone(array2);
  833.         }
  834.         if (array2 == null) {
  835.             return clone(array1);
  836.         }
  837.         final byte[] joinedArray = new byte[array1.length + array2.length];
  838.         System.arraycopy(array1, 0, joinedArray, 0, array1.length);
  839.         System.arraycopy(array2, 0, joinedArray, array1.length, array2.length);
  840.         return joinedArray;
  841.     }

  842.     /**
  843.      * Adds all the elements of the given arrays into a new array.
  844.      * <p>
  845.      * The new array contains all of the element of {@code array1} followed
  846.      * by all of the elements {@code array2}. When an array is returned, it is always
  847.      * a new array.
  848.      * </p>
  849.      * <pre>
  850.      * ArrayUtils.addAll(array1, null)   = cloned copy of array1
  851.      * ArrayUtils.addAll(null, array2)   = cloned copy of array2
  852.      * ArrayUtils.addAll([], [])         = []
  853.      * ArrayUtils.addAll(null, null)     = null
  854.      * </pre>
  855.      *
  856.      * @param array1  the first array whose elements are added to the new array.
  857.      * @param array2  the second array whose elements are added to the new array.
  858.      * @return The new char[] array or {@code null}.
  859.      * @since 2.1
  860.      */
  861.     public static char[] addAll(final char[] array1, final char... array2) {
  862.         if (array1 == null) {
  863.             return clone(array2);
  864.         }
  865.         if (array2 == null) {
  866.             return clone(array1);
  867.         }
  868.         final char[] joinedArray = new char[array1.length + array2.length];
  869.         System.arraycopy(array1, 0, joinedArray, 0, array1.length);
  870.         System.arraycopy(array2, 0, joinedArray, array1.length, array2.length);
  871.         return joinedArray;
  872.     }

  873.     /**
  874.      * Adds all the elements of the given arrays into a new array.
  875.      * <p>
  876.      * The new array contains all of the element of {@code array1} followed
  877.      * by all of the elements {@code array2}. When an array is returned, it is always
  878.      * a new array.
  879.      * </p>
  880.      * <pre>
  881.      * ArrayUtils.addAll(array1, null)   = cloned copy of array1
  882.      * ArrayUtils.addAll(null, array2)   = cloned copy of array2
  883.      * ArrayUtils.addAll([], [])         = []
  884.      * ArrayUtils.addAll(null, null)     = null
  885.      * </pre>
  886.      *
  887.      * @param array1  the first array whose elements are added to the new array.
  888.      * @param array2  the second array whose elements are added to the new array.
  889.      * @return The new double[] array or {@code null}.
  890.      * @since 2.1
  891.      */
  892.     public static double[] addAll(final double[] array1, final double... array2) {
  893.         if (array1 == null) {
  894.             return clone(array2);
  895.         }
  896.         if (array2 == null) {
  897.             return clone(array1);
  898.         }
  899.         final double[] joinedArray = new double[array1.length + array2.length];
  900.         System.arraycopy(array1, 0, joinedArray, 0, array1.length);
  901.         System.arraycopy(array2, 0, joinedArray, array1.length, array2.length);
  902.         return joinedArray;
  903.     }

  904.     /**
  905.      * Adds all the elements of the given arrays into a new array.
  906.      * <p>
  907.      * The new array contains all of the element of {@code array1} followed
  908.      * by all of the elements {@code array2}. When an array is returned, it is always
  909.      * a new array.
  910.      * </p>
  911.      * <pre>
  912.      * ArrayUtils.addAll(array1, null)   = cloned copy of array1
  913.      * ArrayUtils.addAll(null, array2)   = cloned copy of array2
  914.      * ArrayUtils.addAll([], [])         = []
  915.      * ArrayUtils.addAll(null, null)     = null
  916.      * </pre>
  917.      *
  918.      * @param array1  the first array whose elements are added to the new array.
  919.      * @param array2  the second array whose elements are added to the new array.
  920.      * @return The new float[] array or {@code null}.
  921.      * @since 2.1
  922.      */
  923.     public static float[] addAll(final float[] array1, final float... array2) {
  924.         if (array1 == null) {
  925.             return clone(array2);
  926.         }
  927.         if (array2 == null) {
  928.             return clone(array1);
  929.         }
  930.         final float[] joinedArray = new float[array1.length + array2.length];
  931.         System.arraycopy(array1, 0, joinedArray, 0, array1.length);
  932.         System.arraycopy(array2, 0, joinedArray, array1.length, array2.length);
  933.         return joinedArray;
  934.     }

  935.     /**
  936.      * Adds all the elements of the given arrays into a new array.
  937.      * <p>
  938.      * The new array contains all of the element of {@code array1} followed
  939.      * by all of the elements {@code array2}. When an array is returned, it is always
  940.      * a new array.
  941.      * </p>
  942.      * <pre>
  943.      * ArrayUtils.addAll(array1, null)   = cloned copy of array1
  944.      * ArrayUtils.addAll(null, array2)   = cloned copy of array2
  945.      * ArrayUtils.addAll([], [])         = []
  946.      * ArrayUtils.addAll(null, null)     = null
  947.      * </pre>
  948.      *
  949.      * @param array1  the first array whose elements are added to the new array.
  950.      * @param array2  the second array whose elements are added to the new array.
  951.      * @return The new int[] array or {@code null}.
  952.      * @since 2.1
  953.      */
  954.     public static int[] addAll(final int[] array1, final int... array2) {
  955.         if (array1 == null) {
  956.             return clone(array2);
  957.         }
  958.         if (array2 == null) {
  959.             return clone(array1);
  960.         }
  961.         final int[] joinedArray = new int[array1.length + array2.length];
  962.         System.arraycopy(array1, 0, joinedArray, 0, array1.length);
  963.         System.arraycopy(array2, 0, joinedArray, array1.length, array2.length);
  964.         return joinedArray;
  965.     }

  966.     /**
  967.      * Adds all the elements of the given arrays into a new array.
  968.      * <p>
  969.      * The new array contains all of the element of {@code array1} followed
  970.      * by all of the elements {@code array2}. When an array is returned, it is always
  971.      * a new array.
  972.      * </p>
  973.      * <pre>
  974.      * ArrayUtils.addAll(array1, null)   = cloned copy of array1
  975.      * ArrayUtils.addAll(null, array2)   = cloned copy of array2
  976.      * ArrayUtils.addAll([], [])         = []
  977.      * ArrayUtils.addAll(null, null)     = null
  978.      * </pre>
  979.      *
  980.      * @param array1  the first array whose elements are added to the new array.
  981.      * @param array2  the second array whose elements are added to the new array.
  982.      * @return The new long[] array or {@code null}.
  983.      * @since 2.1
  984.      */
  985.     public static long[] addAll(final long[] array1, final long... array2) {
  986.         if (array1 == null) {
  987.             return clone(array2);
  988.         }
  989.         if (array2 == null) {
  990.             return clone(array1);
  991.         }
  992.         final long[] joinedArray = new long[array1.length + array2.length];
  993.         System.arraycopy(array1, 0, joinedArray, 0, array1.length);
  994.         System.arraycopy(array2, 0, joinedArray, array1.length, array2.length);
  995.         return joinedArray;
  996.     }

  997.     /**
  998.      * Adds all the elements of the given arrays into a new array.
  999.      * <p>
  1000.      * The new array contains all of the element of {@code array1} followed
  1001.      * by all of the elements {@code array2}. When an array is returned, it is always
  1002.      * a new array.
  1003.      * </p>
  1004.      * <pre>
  1005.      * ArrayUtils.addAll(array1, null)   = cloned copy of array1
  1006.      * ArrayUtils.addAll(null, array2)   = cloned copy of array2
  1007.      * ArrayUtils.addAll([], [])         = []
  1008.      * ArrayUtils.addAll(null, null)     = null
  1009.      * </pre>
  1010.      *
  1011.      * @param array1  the first array whose elements are added to the new array.
  1012.      * @param array2  the second array whose elements are added to the new array.
  1013.      * @return The new short[] array or {@code null}.
  1014.      * @since 2.1
  1015.      */
  1016.     public static short[] addAll(final short[] array1, final short... array2) {
  1017.         if (array1 == null) {
  1018.             return clone(array2);
  1019.         }
  1020.         if (array2 == null) {
  1021.             return clone(array1);
  1022.         }
  1023.         final short[] joinedArray = new short[array1.length + array2.length];
  1024.         System.arraycopy(array1, 0, joinedArray, 0, array1.length);
  1025.         System.arraycopy(array2, 0, joinedArray, array1.length, array2.length);
  1026.         return joinedArray;
  1027.     }

  1028.     /**
  1029.      * Adds all the elements of the given arrays into a new array.
  1030.      * <p>
  1031.      * The new array contains all of the element of {@code array1} followed
  1032.      * by all of the elements {@code array2}. When an array is returned, it is always
  1033.      * a new array.
  1034.      * </p>
  1035.      * <pre>
  1036.      * ArrayUtils.addAll(null, null)     = null
  1037.      * ArrayUtils.addAll(array1, null)   = cloned copy of array1
  1038.      * ArrayUtils.addAll(null, array2)   = cloned copy of array2
  1039.      * ArrayUtils.addAll([], [])         = []
  1040.      * ArrayUtils.addAll(null, null)     = null
  1041.      * ArrayUtils.addAll([null], [null]) = [null, null]
  1042.      * ArrayUtils.addAll(["a", "b", "c"], ["1", "2", "3"]) = ["a", "b", "c", "1", "2", "3"]
  1043.      * </pre>
  1044.      *
  1045.      * @param <T> the component type of the array
  1046.      * @param array1  the first array whose elements are added to the new array, may be {@code null}
  1047.      * @param array2  the second array whose elements are added to the new array, may be {@code null}
  1048.      * @return The new array, {@code null} if both arrays are {@code null}.
  1049.      *      The type of the new array is the type of the first array,
  1050.      *      unless the first array is null, in which case the type is the same as the second array.
  1051.      * @since 2.1
  1052.      * @throws IllegalArgumentException if the array types are incompatible
  1053.      */
  1054.     public static <T> T[] addAll(final T[] array1, @SuppressWarnings("unchecked") final T... array2) {
  1055.         if (array1 == null) {
  1056.             return clone(array2);
  1057.         }
  1058.         if (array2 == null) {
  1059.             return clone(array1);
  1060.         }
  1061.         final Class<T> type1 = getComponentType(array1);
  1062.         final T[] joinedArray = arraycopy(array1, 0, 0, array1.length, () -> newInstance(type1, array1.length + array2.length));
  1063.         try {
  1064.             System.arraycopy(array2, 0, joinedArray, array1.length, array2.length);
  1065.         } catch (final ArrayStoreException ase) {
  1066.             // Check if problem was due to incompatible types
  1067.             /*
  1068.              * We do this here, rather than before the copy because: - it would be a wasted check most of the time - safer, in case check turns out to be too
  1069.              * strict
  1070.              */
  1071.             final Class<?> type2 = array2.getClass().getComponentType();
  1072.             if (!type1.isAssignableFrom(type2)) {
  1073.                 throw new IllegalArgumentException("Cannot store " + type2.getName() + " in an array of " + type1.getName(), ase);
  1074.             }
  1075.             throw ase; // No, so rethrow original
  1076.         }
  1077.         return joinedArray;
  1078.     }

  1079.     /**
  1080.      * Copies the given array and adds the given element at the beginning of the new array.
  1081.      * <p>
  1082.      * The new array contains the same elements of the input array plus the given element in the first position. The
  1083.      * component type of the new array is the same as that of the input array.
  1084.      * </p>
  1085.      * <p>
  1086.      * If the input array is {@code null}, a new one element array is returned whose component type is the same as the
  1087.      * element.
  1088.      * </p>
  1089.      * <pre>
  1090.      * ArrayUtils.addFirst(null, true)          = [true]
  1091.      * ArrayUtils.addFirst([true], false)       = [false, true]
  1092.      * ArrayUtils.addFirst([true, false], true) = [true, true, false]
  1093.      * </pre>
  1094.      *
  1095.      * @param array the array to "add" the element to, may be {@code null}.
  1096.      * @param element the object to add.
  1097.      * @return A new array containing the existing elements plus the new element The returned array type will be that of
  1098.      *         the input array (unless null), in which case it will have the same type as the element.
  1099.      * @since 3.10
  1100.      */
  1101.     public static boolean[] addFirst(final boolean[] array, final boolean element) {
  1102.         return array == null ? add(array, element) : insert(0, array, element);
  1103.     }

  1104.     /**
  1105.      * Copies the given array and adds the given element at the beginning of the new array.
  1106.      * <p>
  1107.      * The new array contains the same elements of the input array plus the given element in the first position. The
  1108.      * component type of the new array is the same as that of the input array.
  1109.      * </p>
  1110.      * <p>
  1111.      * If the input array is {@code null}, a new one element array is returned whose component type is the same as the
  1112.      * element.
  1113.      * </p>
  1114.      * <pre>
  1115.      * ArrayUtils.addFirst(null, 1)   = [1]
  1116.      * ArrayUtils.addFirst([1], 0)    = [0, 1]
  1117.      * ArrayUtils.addFirst([1, 0], 1) = [1, 1, 0]
  1118.      * </pre>
  1119.      *
  1120.      * @param array the array to "add" the element to, may be {@code null}.
  1121.      * @param element the object to add.
  1122.      * @return A new array containing the existing elements plus the new element The returned array type will be that of
  1123.      *         the input array (unless null), in which case it will have the same type as the element.
  1124.      * @since 3.10
  1125.      */
  1126.     public static byte[] addFirst(final byte[] array, final byte element) {
  1127.         return array == null ? add(array, element) : insert(0, array, element);
  1128.     }

  1129.     /**
  1130.      * Copies the given array and adds the given element at the beginning of the new array.
  1131.      * <p>
  1132.      * The new array contains the same elements of the input array plus the given element in the first position. The
  1133.      * component type of the new array is the same as that of the input array.
  1134.      * </p>
  1135.      * <p>
  1136.      * If the input array is {@code null}, a new one element array is returned whose component type is the same as the
  1137.      * element.
  1138.      * </p>
  1139.      * <pre>
  1140.      * ArrayUtils.addFirst(null, '1')       = ['1']
  1141.      * ArrayUtils.addFirst(['1'], '0')      = ['0', '1']
  1142.      * ArrayUtils.addFirst(['1', '0'], '1') = ['1', '1', '0']
  1143.      * </pre>
  1144.      *
  1145.      * @param array the array to "add" the element to, may be {@code null}.
  1146.      * @param element the object to add.
  1147.      * @return A new array containing the existing elements plus the new element The returned array type will be that of
  1148.      *         the input array (unless null), in which case it will have the same type as the element.
  1149.      * @since 3.10
  1150.      */
  1151.     public static char[] addFirst(final char[] array, final char element) {
  1152.         return array == null ? add(array, element) : insert(0, array, element);
  1153.     }

  1154.     /**
  1155.      * Copies the given array and adds the given element at the beginning of the new array.
  1156.      * <p>
  1157.      * The new array contains the same elements of the input array plus the given element in the first position. The
  1158.      * component type of the new array is the same as that of the input array.
  1159.      * </p>
  1160.      * <p>
  1161.      * If the input array is {@code null}, a new one element array is returned whose component type is the same as the
  1162.      * element.
  1163.      * </p>
  1164.      * <pre>
  1165.      * ArrayUtils.addFirst(null, 1)   = [1]
  1166.      * ArrayUtils.addFirst([1], 0)    = [0, 1]
  1167.      * ArrayUtils.addFirst([1, 0], 1) = [1, 1, 0]
  1168.      * </pre>
  1169.      *
  1170.      * @param array the array to "add" the element to, may be {@code null}.
  1171.      * @param element the object to add.
  1172.      * @return A new array containing the existing elements plus the new element The returned array type will be that of
  1173.      *         the input array (unless null), in which case it will have the same type as the element.
  1174.      * @since 3.10
  1175.      */
  1176.     public static double[] addFirst(final double[] array, final double element) {
  1177.         return array == null ? add(array, element) : insert(0, array, element);
  1178.     }

  1179.     /**
  1180.      * Copies the given array and adds the given element at the beginning of the new array.
  1181.      * <p>
  1182.      * The new array contains the same elements of the input array plus the given element in the first position. The
  1183.      * component type of the new array is the same as that of the input array.
  1184.      * </p>
  1185.      * <p>
  1186.      * If the input array is {@code null}, a new one element array is returned whose component type is the same as the
  1187.      * element.
  1188.      * </p>
  1189.      * <pre>
  1190.      * ArrayUtils.addFirst(null, 1)   = [1]
  1191.      * ArrayUtils.addFirst([1], 0)    = [0, 1]
  1192.      * ArrayUtils.addFirst([1, 0], 1) = [1, 1, 0]
  1193.      * </pre>
  1194.      *
  1195.      * @param array the array to "add" the element to, may be {@code null}.
  1196.      * @param element the object to add.
  1197.      * @return A new array containing the existing elements plus the new element The returned array type will be that of
  1198.      *         the input array (unless null), in which case it will have the same type as the element.
  1199.      * @since 3.10
  1200.      */
  1201.     public static float[] addFirst(final float[] array, final float element) {
  1202.         return array == null ? add(array, element) : insert(0, array, element);
  1203.     }

  1204.     /**
  1205.      * Copies the given array and adds the given element at the beginning of the new array.
  1206.      * <p>
  1207.      * The new array contains the same elements of the input array plus the given element in the first position. The
  1208.      * component type of the new array is the same as that of the input array.
  1209.      * </p>
  1210.      * <p>
  1211.      * If the input array is {@code null}, a new one element array is returned whose component type is the same as the
  1212.      * element.
  1213.      * </p>
  1214.      * <pre>
  1215.      * ArrayUtils.addFirst(null, 1)   = [1]
  1216.      * ArrayUtils.addFirst([1], 0)    = [0, 1]
  1217.      * ArrayUtils.addFirst([1, 0], 1) = [1, 1, 0]
  1218.      * </pre>
  1219.      *
  1220.      * @param array the array to "add" the element to, may be {@code null}.
  1221.      * @param element the object to add.
  1222.      * @return A new array containing the existing elements plus the new element The returned array type will be that of
  1223.      *         the input array (unless null), in which case it will have the same type as the element.
  1224.      * @since 3.10
  1225.      */
  1226.     public static int[] addFirst(final int[] array, final int element) {
  1227.         return array == null ? add(array, element) : insert(0, array, element);
  1228.     }

  1229.     /**
  1230.      * Copies the given array and adds the given element at the beginning of the new array.
  1231.      * <p>
  1232.      * The new array contains the same elements of the input array plus the given element in the first position. The
  1233.      * component type of the new array is the same as that of the input array.
  1234.      * </p>
  1235.      * <p>
  1236.      * If the input array is {@code null}, a new one element array is returned whose component type is the same as the
  1237.      * element.
  1238.      * </p>
  1239.      * <pre>
  1240.      * ArrayUtils.addFirst(null, 1)   = [1]
  1241.      * ArrayUtils.addFirst([1], 0)    = [0, 1]
  1242.      * ArrayUtils.addFirst([1, 0], 1) = [1, 1, 0]
  1243.      * </pre>
  1244.      *
  1245.      * @param array the array to "add" the element to, may be {@code null}.
  1246.      * @param element the object to add.
  1247.      * @return A new array containing the existing elements plus the new element The returned array type will be that of
  1248.      *         the input array (unless null), in which case it will have the same type as the element.
  1249.      * @since 3.10
  1250.      */
  1251.     public static long[] addFirst(final long[] array, final long element) {
  1252.         return array == null ? add(array, element) : insert(0, array, element);
  1253.     }

  1254.     /**
  1255.      * Copies the given array and adds the given element at the beginning of the new array.
  1256.      * <p>
  1257.      * The new array contains the same elements of the input array plus the given element in the first position. The
  1258.      * component type of the new array is the same as that of the input array.
  1259.      * </p>
  1260.      * <p>
  1261.      * If the input array is {@code null}, a new one element array is returned whose component type is the same as the
  1262.      * element.
  1263.      * </p>
  1264.      * <pre>
  1265.      * ArrayUtils.addFirst(null, 1)   = [1]
  1266.      * ArrayUtils.addFirst([1], 0)    = [0, 1]
  1267.      * ArrayUtils.addFirst([1, 0], 1) = [1, 1, 0]
  1268.      * </pre>
  1269.      *
  1270.      * @param array the array to "add" the element to, may be {@code null}.
  1271.      * @param element the object to add.
  1272.      * @return A new array containing the existing elements plus the new element The returned array type will be that of
  1273.      *         the input array (unless null), in which case it will have the same type as the element.
  1274.      * @since 3.10
  1275.      */
  1276.     public static short[] addFirst(final short[] array, final short element) {
  1277.         return array == null ? add(array, element) : insert(0, array, element);
  1278.     }

  1279.     /**
  1280.      * Copies the given array and adds the given element at the beginning of the new array.
  1281.      * <p>
  1282.      * The new array contains the same elements of the input array plus the given element in the first position. The
  1283.      * component type of the new array is the same as that of the input array.
  1284.      * </p>
  1285.      * <p>
  1286.      * If the input array is {@code null}, a new one element array is returned whose component type is the same as the
  1287.      * element, unless the element itself is null, in which case the return type is Object[]
  1288.      * </p>
  1289.      * <pre>
  1290.      * ArrayUtils.addFirst(null, null)      = IllegalArgumentException
  1291.      * ArrayUtils.addFirst(null, "a")       = ["a"]
  1292.      * ArrayUtils.addFirst(["a"], null)     = [null, "a"]
  1293.      * ArrayUtils.addFirst(["a"], "b")      = ["b", "a"]
  1294.      * ArrayUtils.addFirst(["a", "b"], "c") = ["c", "a", "b"]
  1295.      * </pre>
  1296.      *
  1297.      * @param <T> the component type of the array
  1298.      * @param array the array to "add" the element to, may be {@code null}
  1299.      * @param element the object to add, may be {@code null}
  1300.      * @return A new array containing the existing elements plus the new element The returned array type will be that of
  1301.      *         the input array (unless null), in which case it will have the same type as the element. If both are null,
  1302.      *         an IllegalArgumentException is thrown
  1303.      * @since 3.10
  1304.      * @throws IllegalArgumentException if both arguments are null
  1305.      */
  1306.     public static <T> T[] addFirst(final T[] array, final T element) {
  1307.         return array == null ? add(array, element) : insert(0, array, element);
  1308.     }

  1309.     /**
  1310.      * A fluent version of {@link System#arraycopy(Object, int, Object, int, int)} that returns the destination array.
  1311.      *
  1312.      * @param <T>       the type.
  1313.      * @param source    the source array.
  1314.      * @param sourcePos starting position in the source array.
  1315.      * @param destPos   starting position in the destination data.
  1316.      * @param length    the number of array elements to be copied.
  1317.      * @param allocator allocates the array to populate and return.
  1318.      * @return dest
  1319.      * @throws IndexOutOfBoundsException if copying would cause access of data outside array bounds.
  1320.      * @throws ArrayStoreException       if an element in the {@code src} array could not be stored into the {@code dest} array because of a type
  1321.      *                                   mismatch.
  1322.      * @throws NullPointerException      if either {@code src} or {@code dest} is {@code null}.
  1323.      * @since 3.15.0
  1324.      */
  1325.     public static <T> T arraycopy(final T source, final int sourcePos, final int destPos, final int length, final Function<Integer, T> allocator) {
  1326.         return arraycopy(source, sourcePos, allocator.apply(length), destPos, length);
  1327.     }

  1328.     /**
  1329.      * A fluent version of {@link System#arraycopy(Object, int, Object, int, int)} that returns the destination array.
  1330.      *
  1331.      * @param <T>       the type.
  1332.      * @param source    the source array.
  1333.      * @param sourcePos starting position in the source array.
  1334.      * @param destPos   starting position in the destination data.
  1335.      * @param length    the number of array elements to be copied.
  1336.      * @param allocator allocates the array to populate and return.
  1337.      * @return dest
  1338.      * @throws IndexOutOfBoundsException if copying would cause access of data outside array bounds.
  1339.      * @throws ArrayStoreException       if an element in the {@code src} array could not be stored into the {@code dest} array because of a type
  1340.      *                                   mismatch.
  1341.      * @throws NullPointerException      if either {@code src} or {@code dest} is {@code null}.
  1342.      * @since 3.15.0
  1343.      */
  1344.     public static <T> T arraycopy(final T source, final int sourcePos, final int destPos, final int length, final Supplier<T> allocator) {
  1345.         return arraycopy(source, sourcePos, allocator.get(), destPos, length);
  1346.     }

  1347.     /**
  1348.      * A fluent version of {@link System#arraycopy(Object, int, Object, int, int)} that returns the destination array.
  1349.      *
  1350.      * @param <T>       the type
  1351.      * @param source    the source array.
  1352.      * @param sourcePos starting position in the source array.
  1353.      * @param dest      the destination array.
  1354.      * @param destPos   starting position in the destination data.
  1355.      * @param length    the number of array elements to be copied.
  1356.      * @return dest
  1357.      * @throws IndexOutOfBoundsException if copying would cause access of data outside array bounds.
  1358.      * @throws ArrayStoreException       if an element in the {@code src} array could not be stored into the {@code dest} array because of a type
  1359.      *                                   mismatch.
  1360.      * @throws NullPointerException      if either {@code src} or {@code dest} is {@code null}.
  1361.      * @since 3.15.0
  1362.      */
  1363.     public static <T> T arraycopy(final T source, final int sourcePos, final T dest, final int destPos, final int length) {
  1364.         System.arraycopy(source, sourcePos, dest, destPos, length);
  1365.         return dest;
  1366.     }

  1367.     /**
  1368.      * Clones an array or returns {@code null}.
  1369.      * <p>
  1370.      * This method returns {@code null} for a {@code null} input array.
  1371.      * </p>
  1372.      *
  1373.      * @param array the array to clone, may be {@code null}
  1374.      * @return the cloned array, {@code null} if {@code null} input
  1375.      */
  1376.     public static boolean[] clone(final boolean[] array) {
  1377.         return array != null ? array.clone() : null;
  1378.     }

  1379.     /**
  1380.      * Clones an array or returns {@code null}.
  1381.      * <p>
  1382.      * This method returns {@code null} for a {@code null} input array.
  1383.      * </p>
  1384.      *
  1385.      * @param array the array to clone, may be {@code null}
  1386.      * @return the cloned array, {@code null} if {@code null} input
  1387.      */
  1388.     public static byte[] clone(final byte[] array) {
  1389.         return array != null ? array.clone() : null;
  1390.     }

  1391.     /**
  1392.      * Clones an array or returns {@code null}.
  1393.      * <p>
  1394.      * This method returns {@code null} for a {@code null} input array.
  1395.      * </p>
  1396.      *
  1397.      * @param array the array to clone, may be {@code null}
  1398.      * @return the cloned array, {@code null} if {@code null} input
  1399.      */
  1400.     public static char[] clone(final char[] array) {
  1401.         return array != null ? array.clone() : null;
  1402.     }

  1403.     /**
  1404.      * Clones an array or returns {@code null}.
  1405.      * <p>
  1406.      * This method returns {@code null} for a {@code null} input array.
  1407.      * </p>
  1408.      *
  1409.      * @param array the array to clone, may be {@code null}
  1410.      * @return the cloned array, {@code null} if {@code null} input
  1411.      */
  1412.     public static double[] clone(final double[] array) {
  1413.         return array != null ? array.clone() : null;
  1414.     }

  1415.     /**
  1416.      * Clones an array or returns {@code null}.
  1417.      * <p>
  1418.      * This method returns {@code null} for a {@code null} input array.
  1419.      * </p>
  1420.      *
  1421.      * @param array the array to clone, may be {@code null}
  1422.      * @return the cloned array, {@code null} if {@code null} input
  1423.      */
  1424.     public static float[] clone(final float[] array) {
  1425.         return array != null ? array.clone() : null;
  1426.     }

  1427.     /**
  1428.      * Clones an array or returns {@code null}.
  1429.      * <p>
  1430.      * This method returns {@code null} for a {@code null} input array.
  1431.      * </p>
  1432.      *
  1433.      * @param array the array to clone, may be {@code null}
  1434.      * @return the cloned array, {@code null} if {@code null} input
  1435.      */
  1436.     public static int[] clone(final int[] array) {
  1437.         return array != null ? array.clone() : null;
  1438.     }

  1439.     /**
  1440.      * Clones an array or returns {@code null}.
  1441.      * <p>
  1442.      * This method returns {@code null} for a {@code null} input array.
  1443.      * </p>
  1444.      *
  1445.      * @param array the array to clone, may be {@code null}
  1446.      * @return the cloned array, {@code null} if {@code null} input
  1447.      */
  1448.     public static long[] clone(final long[] array) {
  1449.         return array != null ? array.clone() : null;
  1450.     }

  1451.     /**
  1452.      * Clones an array or returns {@code null}.
  1453.      * <p>
  1454.      * This method returns {@code null} for a {@code null} input array.
  1455.      * </p>
  1456.      *
  1457.      * @param array the array to clone, may be {@code null}
  1458.      * @return the cloned array, {@code null} if {@code null} input
  1459.      */
  1460.     public static short[] clone(final short[] array) {
  1461.         return array != null ? array.clone() : null;
  1462.     }

  1463.     /**
  1464.      * Shallow clones an array or returns {@code null}.
  1465.      * <p>
  1466.      * The objects in the array are not cloned, thus there is no special handling for multi-dimensional arrays.
  1467.      * </p>
  1468.      * <p>
  1469.      * This method returns {@code null} for a {@code null} input array.
  1470.      * </p>
  1471.      *
  1472.      * @param <T>   the component type of the array
  1473.      * @param array the array to shallow clone, may be {@code null}
  1474.      * @return the cloned array, {@code null} if {@code null} input
  1475.      */
  1476.     public static <T> T[] clone(final T[] array) {
  1477.         return array != null ? array.clone() : null;
  1478.     }

  1479.     /**
  1480.      * Checks if the value is in the given array.
  1481.      * <p>
  1482.      * The method returns {@code false} if a {@code null} array is passed in.
  1483.      * </p>
  1484.      *
  1485.      * @param array  the array to search through
  1486.      * @param valueToFind  the value to find
  1487.      * @return {@code true} if the array contains the object
  1488.      */
  1489.     public static boolean contains(final boolean[] array, final boolean valueToFind) {
  1490.         return indexOf(array, valueToFind) != INDEX_NOT_FOUND;
  1491.     }

  1492.     /**
  1493.      * Checks if the value is in the given array.
  1494.      * <p>
  1495.      * The method returns {@code false} if a {@code null} array is passed in.
  1496.      * </p>
  1497.      *
  1498.      * @param array  the array to search through
  1499.      * @param valueToFind  the value to find
  1500.      * @return {@code true} if the array contains the object
  1501.      */
  1502.     public static boolean contains(final byte[] array, final byte valueToFind) {
  1503.         return indexOf(array, valueToFind) != INDEX_NOT_FOUND;
  1504.     }

  1505.     /**
  1506.      * Checks if the value is in the given array.
  1507.      * <p>
  1508.      * The method returns {@code false} if a {@code null} array is passed in.
  1509.      * </p>
  1510.      *
  1511.      * @param array  the array to search through
  1512.      * @param valueToFind  the value to find
  1513.      * @return {@code true} if the array contains the object
  1514.      * @since 2.1
  1515.      */
  1516.     public static boolean contains(final char[] array, final char valueToFind) {
  1517.         return indexOf(array, valueToFind) != INDEX_NOT_FOUND;
  1518.     }

  1519.     /**
  1520.      * Checks if the value is in the given array.
  1521.      * <p>
  1522.      * The method returns {@code false} if a {@code null} array is passed in.
  1523.      * </p>
  1524.      *
  1525.      * @param array  the array to search through
  1526.      * @param valueToFind  the value to find
  1527.      * @return {@code true} if the array contains the object
  1528.      */
  1529.     public static boolean contains(final double[] array, final double valueToFind) {
  1530.         return indexOf(array, valueToFind) != INDEX_NOT_FOUND;
  1531.     }

  1532.     /**
  1533.      * Checks if a value falling within the given tolerance is in the
  1534.      * given array.  If the array contains a value within the inclusive range
  1535.      * defined by (value - tolerance) to (value + tolerance).
  1536.      * <p>
  1537.      * The method returns {@code false} if a {@code null} array
  1538.      * is passed in.
  1539.      * </p>
  1540.      *
  1541.      * @param array  the array to search
  1542.      * @param valueToFind  the value to find
  1543.      * @param tolerance  the array contains the tolerance of the search
  1544.      * @return true if value falling within tolerance is in array
  1545.      */
  1546.     public static boolean contains(final double[] array, final double valueToFind, final double tolerance) {
  1547.         return indexOf(array, valueToFind, 0, tolerance) != INDEX_NOT_FOUND;
  1548.     }

  1549.     /**
  1550.      * Checks if the value is in the given array.
  1551.      * <p>
  1552.      * The method returns {@code false} if a {@code null} array is passed in.
  1553.      * </p>
  1554.      *
  1555.      * @param array  the array to search through
  1556.      * @param valueToFind  the value to find
  1557.      * @return {@code true} if the array contains the object
  1558.      */
  1559.     public static boolean contains(final float[] array, final float valueToFind) {
  1560.         return indexOf(array, valueToFind) != INDEX_NOT_FOUND;
  1561.     }

  1562.     /**
  1563.      * Checks if the value is in the given array.
  1564.      * <p>
  1565.      * The method returns {@code false} if a {@code null} array is passed in.
  1566.      * </p>
  1567.      *
  1568.      * @param array  the array to search through
  1569.      * @param valueToFind  the value to find
  1570.      * @return {@code true} if the array contains the object
  1571.      */
  1572.     public static boolean contains(final int[] array, final int valueToFind) {
  1573.         return indexOf(array, valueToFind) != INDEX_NOT_FOUND;
  1574.     }

  1575.     /**
  1576.      * Checks if the value is in the given array.
  1577.      * <p>
  1578.      * The method returns {@code false} if a {@code null} array is passed in.
  1579.      * </p>
  1580.      *
  1581.      * @param array  the array to search through
  1582.      * @param valueToFind  the value to find
  1583.      * @return {@code true} if the array contains the object
  1584.      */
  1585.     public static boolean contains(final long[] array, final long valueToFind) {
  1586.         return indexOf(array, valueToFind) != INDEX_NOT_FOUND;
  1587.     }

  1588.     /**
  1589.      * Checks if the object is in the given array.
  1590.      * <p>
  1591.      * The method returns {@code false} if a {@code null} array is passed in.
  1592.      * </p>
  1593.      *
  1594.      * @param array  the array to search through
  1595.      * @param objectToFind  the object to find
  1596.      * @return {@code true} if the array contains the object
  1597.      */
  1598.     public static boolean contains(final Object[] array, final Object objectToFind) {
  1599.         return indexOf(array, objectToFind) != INDEX_NOT_FOUND;
  1600.     }

  1601.     /**
  1602.      * Checks if the value is in the given array.
  1603.      * <p>
  1604.      * The method returns {@code false} if a {@code null} array is passed in.
  1605.      * </p>
  1606.      *
  1607.      * @param array  the array to search through
  1608.      * @param valueToFind  the value to find
  1609.      * @return {@code true} if the array contains the object
  1610.      */
  1611.     public static boolean contains(final short[] array, final short valueToFind) {
  1612.         return indexOf(array, valueToFind) != INDEX_NOT_FOUND;
  1613.     }

  1614.     /**
  1615.      * Checks if any of the objects are in the given array.
  1616.      * <p>
  1617.      * The method returns {@code false} if a {@code null} array is passed in.
  1618.      * </p>
  1619.      *
  1620.      * @param array  the array to search through
  1621.      * @param objectsToFind  any of the objects to find
  1622.      * @return {@code true} if the array contains any of the objects
  1623.      * @since 3.13.0
  1624.      */
  1625.     public static boolean containsAny(final Object[] array, final Object... objectsToFind) {
  1626.         return Streams.of(objectsToFind).anyMatch(e -> contains(array, e));
  1627.     }

  1628.     /**
  1629.      * Returns a copy of the given array of size 1 greater than the argument.
  1630.      * The last value of the array is left to the default value.
  1631.      *
  1632.      * @param array The array to copy, must not be {@code null}.
  1633.      * @param newArrayComponentType If {@code array} is {@code null}, create a
  1634.      * size 1 array of this type.
  1635.      * @return A new copy of the array of size 1 greater than the input.
  1636.      */
  1637.     private static Object copyArrayGrow1(final Object array, final Class<?> newArrayComponentType) {
  1638.         if (array != null) {
  1639.             final int arrayLength = Array.getLength(array);
  1640.             final Object newArray = Array.newInstance(array.getClass().getComponentType(), arrayLength + 1);
  1641.             System.arraycopy(array, 0, newArray, 0, arrayLength);
  1642.             return newArray;
  1643.         }
  1644.         return Array.newInstance(newArrayComponentType, 1);
  1645.     }

  1646.     /**
  1647.      * Gets the nTh element of an array or null if the index is out of bounds or the array is null.
  1648.      *
  1649.      * @param <T> The type of array elements.
  1650.      * @param array The array to index.
  1651.      * @param index The index
  1652.      * @return the nTh element of an array or null if the index is out of bounds or the array is null.
  1653.      * @since 3.11
  1654.      */
  1655.     public static <T> T get(final T[] array, final int index) {
  1656.         return get(array, index, null);
  1657.     }

  1658.     /**
  1659.      * Gets the nTh element of an array or a default value if the index is out of bounds.
  1660.      *
  1661.      * @param <T> The type of array elements.
  1662.      * @param array The array to index.
  1663.      * @param index The index
  1664.      * @param defaultValue The return value of the given index is out of bounds.
  1665.      * @return the nTh element of an array or a default value if the index is out of bounds.
  1666.      * @since 3.11
  1667.      */
  1668.     public static <T> T get(final T[] array, final int index, final T defaultValue) {
  1669.         return isArrayIndexValid(array, index) ? array[index] : defaultValue;
  1670.     }

  1671.     /**
  1672.      * Gets an array's component type.
  1673.      *
  1674.      * @param <T> The array type.
  1675.      * @param array The array.
  1676.      * @return The component type.
  1677.      * @since 3.13.0
  1678.      */
  1679.     public static <T> Class<T> getComponentType(final T[] array) {
  1680.         return ClassUtils.getComponentType(ObjectUtils.getClass(array));
  1681.     }

  1682.     /**
  1683.      * Returns the length of the specified array.
  1684.      * This method can deal with {@link Object} arrays and with primitive arrays.
  1685.      * <p>
  1686.      * If the input array is {@code null}, {@code 0} is returned.
  1687.      * </p>
  1688.      * <pre>
  1689.      * ArrayUtils.getLength(null)            = 0
  1690.      * ArrayUtils.getLength([])              = 0
  1691.      * ArrayUtils.getLength([null])          = 1
  1692.      * ArrayUtils.getLength([true, false])   = 2
  1693.      * ArrayUtils.getLength([1, 2, 3])       = 3
  1694.      * ArrayUtils.getLength(["a", "b", "c"]) = 3
  1695.      * </pre>
  1696.      *
  1697.      * @param array  the array to retrieve the length from, may be null
  1698.      * @return The length of the array, or {@code 0} if the array is {@code null}
  1699.      * @throws IllegalArgumentException if the object argument is not an array.
  1700.      * @since 2.1
  1701.      */
  1702.     public static int getLength(final Object array) {
  1703.         return array != null ? Array.getLength(array) : 0;
  1704.     }

  1705.     /**
  1706.      * Gets a hash code for an array handling multidimensional arrays correctly.
  1707.      * <p>
  1708.      * Multi-dimensional primitive arrays are also handled correctly by this method.
  1709.      * </p>
  1710.      *
  1711.      * @param array  the array to get a hash code for, {@code null} returns zero
  1712.      * @return a hash code for the array
  1713.      */
  1714.     public static int hashCode(final Object array) {
  1715.         return new HashCodeBuilder().append(array).toHashCode();
  1716.     }

  1717.     /**
  1718.      * Finds the indices of the given value in the array.
  1719.      * <p>
  1720.      * This method returns an empty BitSet for a {@code null} input array.
  1721.      * </p>
  1722.      *
  1723.      * @param array  the array to search through for the object, may be {@code null}
  1724.      * @param valueToFind  the value to find
  1725.      * @return a BitSet of all the indices of the value within the array,
  1726.      *  an empty BitSet if not found or {@code null} array input
  1727.      * @since 3.10
  1728.      */
  1729.     public static BitSet indexesOf(final boolean[] array, final boolean valueToFind) {
  1730.         return indexesOf(array, valueToFind, 0);
  1731.     }

  1732.     /**
  1733.      * Finds the indices of the given value in the array starting at the given index.
  1734.      * <p>
  1735.      * This method returns an empty BitSet for a {@code null} input array.
  1736.      * </p>
  1737.      * <p>
  1738.      * A negative startIndex is treated as zero. A startIndex larger than the array
  1739.      * length will return an empty BitSet ({@code -1}).
  1740.      * </p>
  1741.      *
  1742.      * @param array  the array to search through for the object, may be {@code null}
  1743.      * @param valueToFind  the value to find
  1744.      * @param startIndex  the index to start searching at
  1745.      * @return a BitSet of all the indices of the value within the array,
  1746.      *  an empty BitSet if not found or {@code null}
  1747.      *  array input
  1748.      * @since 3.10
  1749.      */
  1750.     public static BitSet indexesOf(final boolean[] array, final boolean valueToFind, int startIndex) {
  1751.         final BitSet bitSet = new BitSet();
  1752.         if (array == null) {
  1753.             return bitSet;
  1754.         }
  1755.         while (startIndex < array.length) {
  1756.             startIndex = indexOf(array, valueToFind, startIndex);
  1757.             if (startIndex == INDEX_NOT_FOUND) {
  1758.                 break;
  1759.             }
  1760.             bitSet.set(startIndex);
  1761.             ++startIndex;
  1762.         }
  1763.         return bitSet;
  1764.     }

  1765.     /**
  1766.      * Finds the indices of the given value in the array.
  1767.      *
  1768.      * <p>This method returns an empty BitSet for a {@code null} input array.</p>
  1769.      *
  1770.      * @param array  the array to search through for the object, may be {@code null}
  1771.      * @param valueToFind  the value to find
  1772.      * @return a BitSet of all the indices of the value within the array,
  1773.      *  an empty BitSet if not found or {@code null} array input
  1774.      * @since 3.10
  1775.      */
  1776.     public static BitSet indexesOf(final byte[] array, final byte valueToFind) {
  1777.         return indexesOf(array, valueToFind, 0);
  1778.     }

  1779.     /**
  1780.      * Finds the indices of the given value in the array starting at the given index.
  1781.      *
  1782.      * <p>This method returns an empty BitSet for a {@code null} input array.</p>
  1783.      *
  1784.      * <p>A negative startIndex is treated as zero. A startIndex larger than the array
  1785.      * length will return an empty BitSet.</p>
  1786.      *
  1787.      * @param array  the array to search through for the object, may be {@code null}
  1788.      * @param valueToFind  the value to find
  1789.      * @param startIndex  the index to start searching at
  1790.      * @return a BitSet of all the indices of the value within the array,
  1791.      *  an empty BitSet if not found or {@code null} array input
  1792.      * @since 3.10
  1793.      */
  1794.     public static BitSet indexesOf(final byte[] array, final byte valueToFind, int startIndex) {
  1795.         final BitSet bitSet = new BitSet();
  1796.         if (array == null) {
  1797.             return bitSet;
  1798.         }
  1799.         while (startIndex < array.length) {
  1800.             startIndex = indexOf(array, valueToFind, startIndex);
  1801.             if (startIndex == INDEX_NOT_FOUND) {
  1802.                 break;
  1803.             }
  1804.             bitSet.set(startIndex);
  1805.             ++startIndex;
  1806.         }

  1807.         return bitSet;
  1808.     }

  1809.     /**
  1810.      * Finds the indices of the given value in the array.
  1811.      *
  1812.      * <p>This method returns an empty BitSet for a {@code null} input array.</p>
  1813.      *
  1814.      * @param array  the array to search through for the object, may be {@code null}
  1815.      * @param valueToFind  the value to find
  1816.      * @return a BitSet of all the indices of the value within the array,
  1817.      *  an empty BitSet if not found or {@code null} array input
  1818.      * @since 3.10
  1819.      */
  1820.     public static BitSet indexesOf(final char[] array, final char valueToFind) {
  1821.         return indexesOf(array, valueToFind, 0);
  1822.     }

  1823.     /**
  1824.      * Finds the indices of the given value in the array starting at the given index.
  1825.      *
  1826.      * <p>This method returns an empty BitSet for a {@code null} input array.</p>
  1827.      *
  1828.      * <p>A negative startIndex is treated as zero. A startIndex larger than the array
  1829.      * length will return an empty BitSet.</p>
  1830.      *
  1831.      * @param array  the array to search through for the object, may be {@code null}
  1832.      * @param valueToFind  the value to find
  1833.      * @param startIndex  the index to start searching at
  1834.      * @return a BitSet of all the indices of the value within the array,
  1835.      *  an empty BitSet if not found or {@code null} array input
  1836.      * @since 3.10
  1837.      */
  1838.     public static BitSet indexesOf(final char[] array, final char valueToFind, int startIndex) {
  1839.         final BitSet bitSet = new BitSet();
  1840.         if (array == null) {
  1841.             return bitSet;
  1842.         }
  1843.         while (startIndex < array.length) {
  1844.             startIndex = indexOf(array, valueToFind, startIndex);
  1845.             if (startIndex == INDEX_NOT_FOUND) {
  1846.                 break;
  1847.             }
  1848.             bitSet.set(startIndex);
  1849.             ++startIndex;
  1850.         }
  1851.         return bitSet;
  1852.     }

  1853.     /**
  1854.      * Finds the indices of the given value in the array.
  1855.      *
  1856.      * <p>This method returns empty BitSet for a {@code null} input array.</p>
  1857.      *
  1858.      * @param array  the array to search through for the object, may be {@code null}
  1859.      * @param valueToFind  the value to find
  1860.      * @return a BitSet of all the indices of the value within the array,
  1861.      *  an empty BitSet if not found or {@code null} array input
  1862.      * @since 3.10
  1863.      */
  1864.     public static BitSet indexesOf(final double[] array, final double valueToFind) {
  1865.         return indexesOf(array, valueToFind, 0);
  1866.     }

  1867.     /**
  1868.      * Finds the indices of the given value within a given tolerance in the array.
  1869.      *
  1870.      * <p>
  1871.      * This method will return all the indices of the value which fall between the region
  1872.      * defined by valueToFind - tolerance and valueToFind + tolerance, each time between the nearest integers.
  1873.      * </p>
  1874.      *
  1875.      * <p>This method returns an empty BitSet for a {@code null} input array.</p>
  1876.      *
  1877.      * @param array  the array to search through for the object, may be {@code null}
  1878.      * @param valueToFind  the value to find
  1879.      * @param tolerance tolerance of the search
  1880.      * @return a BitSet of all the indices of the value within the array,
  1881.      *  an empty BitSet if not found or {@code null} array input
  1882.      * @since 3.10
  1883.      */
  1884.     public static BitSet indexesOf(final double[] array, final double valueToFind, final double tolerance) {
  1885.         return indexesOf(array, valueToFind, 0, tolerance);
  1886.     }

  1887.     /**
  1888.      * Finds the indices of the given value in the array starting at the given index.
  1889.      *
  1890.      * <p>This method returns an empty BitSet for a {@code null} input array.</p>
  1891.      *
  1892.      * <p>A negative startIndex is treated as zero. A startIndex larger than the array
  1893.      * length will return an empty BitSet.</p>
  1894.      *
  1895.      * @param array  the array to search through for the object, may be {@code null}
  1896.      * @param valueToFind  the value to find
  1897.      * @param startIndex  the index to start searching at
  1898.      * @return a BitSet of the indices of the value within the array,
  1899.      *  an empty BitSet if not found or {@code null} array input
  1900.      * @since 3.10
  1901.      */
  1902.     public static BitSet indexesOf(final double[] array, final double valueToFind, int startIndex) {
  1903.         final BitSet bitSet = new BitSet();
  1904.         if (array == null) {
  1905.             return bitSet;
  1906.         }
  1907.         while (startIndex < array.length) {
  1908.             startIndex = indexOf(array, valueToFind, startIndex);
  1909.             if (startIndex == INDEX_NOT_FOUND) {
  1910.                 break;
  1911.             }
  1912.             bitSet.set(startIndex);
  1913.             ++startIndex;
  1914.         }
  1915.         return bitSet;
  1916.     }

  1917.     /**
  1918.      * Finds the indices of the given value in the array starting at the given index.
  1919.      *
  1920.      * <p>
  1921.      * This method will return the indices of the values which fall between the region
  1922.      * defined by valueToFind - tolerance and valueToFind + tolerance, between the nearest integers.
  1923.      * </p>
  1924.      *
  1925.      * <p>This method returns an empty BitSet for a {@code null} input array.</p>
  1926.      *
  1927.      * <p>A negative startIndex is treated as zero. A startIndex larger than the array
  1928.      * length will return an empty BitSet.</p>
  1929.      *
  1930.      * @param array  the array to search through for the object, may be {@code null}
  1931.      * @param valueToFind  the value to find
  1932.      * @param startIndex  the index to start searching at
  1933.      * @param tolerance tolerance of the search
  1934.      * @return a BitSet of the indices of the value within the array,
  1935.      *  an empty BitSet if not found or {@code null} array input
  1936.      * @since 3.10
  1937.      */
  1938.     public static BitSet indexesOf(final double[] array, final double valueToFind, int startIndex, final double tolerance) {
  1939.         final BitSet bitSet = new BitSet();
  1940.         if (array == null) {
  1941.             return bitSet;
  1942.         }
  1943.         while (startIndex < array.length) {
  1944.             startIndex = indexOf(array, valueToFind, startIndex, tolerance);
  1945.             if (startIndex == INDEX_NOT_FOUND) {
  1946.                 break;
  1947.             }
  1948.             bitSet.set(startIndex);
  1949.             ++startIndex;
  1950.         }
  1951.         return bitSet;
  1952.     }

  1953.     /**
  1954.      * Finds the indices of the given value in the array.
  1955.      *
  1956.      * <p>This method returns an empty BitSet for a {@code null} input array.</p>
  1957.      *
  1958.      * @param array  the array to search through for the object, may be {@code null}
  1959.      * @param valueToFind  the value to find
  1960.      * @return a BitSet of all the indices of the value within the array,
  1961.      *  an empty BitSet if not found or {@code null} array input
  1962.      * @since 3.10
  1963.      */
  1964.     public static BitSet indexesOf(final float[] array, final float valueToFind) {
  1965.         return indexesOf(array, valueToFind, 0);
  1966.     }

  1967.     /**
  1968.      * Finds the indices of the given value in the array starting at the given index.
  1969.      *
  1970.      * <p>This method returns an empty BitSet for a {@code null} input array.</p>
  1971.      *
  1972.      * <p>A negative startIndex is treated as zero. A startIndex larger than the array
  1973.      * length will return empty BitSet.</p>
  1974.      *
  1975.      * @param array  the array to search through for the object, may be {@code null}
  1976.      * @param valueToFind  the value to find
  1977.      * @param startIndex  the index to start searching at
  1978.      * @return a BitSet of all the indices of the value within the array,
  1979.      *  an empty BitSet if not found or {@code null} array input
  1980.      * @since 3.10
  1981.      */
  1982.     public static BitSet indexesOf(final float[] array, final float valueToFind, int startIndex) {
  1983.         final BitSet bitSet = new BitSet();
  1984.         if (array == null) {
  1985.             return bitSet;
  1986.         }
  1987.         while (startIndex < array.length) {
  1988.             startIndex = indexOf(array, valueToFind, startIndex);
  1989.             if (startIndex == INDEX_NOT_FOUND) {
  1990.                 break;
  1991.             }
  1992.             bitSet.set(startIndex);
  1993.             ++startIndex;
  1994.         }
  1995.         return bitSet;
  1996.     }

  1997.     /**
  1998.      * Finds the indices of the given value in the array.
  1999.      *
  2000.      * <p>This method returns an empty BitSet for a {@code null} input array.</p>
  2001.      *
  2002.      * @param array  the array to search through for the object, may be {@code null}
  2003.      * @param valueToFind  the value to find
  2004.      * @return a BitSet of all the indices of the value within the array,
  2005.      *  an empty BitSet if not found or {@code null} array input
  2006.      * @since 3.10
  2007.      */
  2008.     public static BitSet indexesOf(final int[] array, final int valueToFind) {
  2009.         return indexesOf(array, valueToFind, 0);
  2010.     }

  2011.     /**
  2012.      * Finds the indices of the given value in the array starting at the given index.
  2013.      *
  2014.      * <p>This method returns an empty BitSet for a {@code null} input array.</p>
  2015.      *
  2016.      * <p>A negative startIndex is treated as zero. A startIndex larger than the array
  2017.      * length will return an empty BitSet.</p>
  2018.      *
  2019.      * @param array  the array to search through for the object, may be {@code null}
  2020.      * @param valueToFind  the value to find
  2021.      * @param startIndex  the index to start searching at
  2022.      * @return a BitSet of all the indices of the value within the array,
  2023.      *  an empty BitSet if not found or {@code null} array input
  2024.      * @since 3.10
  2025.      */
  2026.     public static BitSet indexesOf(final int[] array, final int valueToFind, int startIndex) {
  2027.         final BitSet bitSet = new BitSet();
  2028.         if (array == null) {
  2029.             return bitSet;
  2030.         }
  2031.         while (startIndex < array.length) {
  2032.             startIndex = indexOf(array, valueToFind, startIndex);

  2033.             if (startIndex == INDEX_NOT_FOUND) {
  2034.                 break;
  2035.             }
  2036.             bitSet.set(startIndex);
  2037.             ++startIndex;
  2038.         }
  2039.         return bitSet;
  2040.     }

  2041.     /**
  2042.      * Finds the indices of the given value in the array.
  2043.      *
  2044.      * <p>This method returns an empty BitSet for a {@code null} input array.</p>
  2045.      *
  2046.      * @param array  the array to search through for the object, may be {@code null}
  2047.      * @param valueToFind  the value to find
  2048.      * @return a BitSet of all the indices of the value within the array,
  2049.      *  an empty BitSet if not found or {@code null} array input
  2050.      * @since 3.10
  2051.      */
  2052.     public static BitSet indexesOf(final long[] array, final long valueToFind) {
  2053.         return indexesOf(array, valueToFind, 0);
  2054.     }

  2055.     /**
  2056.      * Finds the indices of the given value in the array starting at the given index.
  2057.      *
  2058.      * <p>This method returns an empty BitSet for a {@code null} input array.</p>
  2059.      *
  2060.      * <p>A negative startIndex is treated as zero. A startIndex larger than the array
  2061.      * length will return an empty BitSet.</p>
  2062.      *
  2063.      * @param array  the array to search through for the object, may be {@code null}
  2064.      * @param valueToFind  the value to find
  2065.      * @param startIndex  the index to start searching at
  2066.      * @return a BitSet of all the indices of the value within the array,
  2067.      *  an empty BitSet if not found or {@code null} array input
  2068.      * @since 3.10
  2069.      */
  2070.     public static BitSet indexesOf(final long[] array, final long valueToFind, int startIndex) {
  2071.         final BitSet bitSet = new BitSet();
  2072.         if (array == null) {
  2073.             return bitSet;
  2074.         }
  2075.         while (startIndex < array.length) {
  2076.             startIndex = indexOf(array, valueToFind, startIndex);
  2077.             if (startIndex == INDEX_NOT_FOUND) {
  2078.                 break;
  2079.             }
  2080.             bitSet.set(startIndex);
  2081.             ++startIndex;
  2082.         }
  2083.         return bitSet;
  2084.     }

  2085.     /**
  2086.      * Finds the indices of the given object in the array.
  2087.      *
  2088.      * <p>This method returns an empty BitSet for a {@code null} input array.</p>
  2089.      *
  2090.      * @param array  the array to search through for the object, may be {@code null}
  2091.      * @param objectToFind  the object to find, may be {@code null}
  2092.      * @return a BitSet of all the indices of the object within the array,
  2093.      *  an empty BitSet if not found or {@code null} array input
  2094.      * @since 3.10
  2095.      */
  2096.     public static BitSet indexesOf(final Object[] array, final Object objectToFind) {
  2097.         return indexesOf(array, objectToFind, 0);
  2098.     }

  2099.     /**
  2100.      * Finds the indices of the given object in the array starting at the given index.
  2101.      *
  2102.      * <p>This method returns an empty BitSet for a {@code null} input array.</p>
  2103.      *
  2104.      * <p>A negative startIndex is treated as zero. A startIndex larger than the array
  2105.      * length will return an empty BitSet.</p>
  2106.      *
  2107.      * @param array  the array to search through for the object, may be {@code null}
  2108.      * @param objectToFind  the object to find, may be {@code null}
  2109.      * @param startIndex  the index to start searching at
  2110.      * @return a BitSet of all the indices of the object within the array starting at the index,
  2111.      *  an empty BitSet if not found or {@code null} array input
  2112.      * @since 3.10
  2113.      */
  2114.     public static BitSet indexesOf(final Object[] array, final Object objectToFind, int startIndex) {
  2115.         final BitSet bitSet = new BitSet();
  2116.         if (array == null) {
  2117.             return bitSet;
  2118.         }
  2119.         while (startIndex < array.length) {
  2120.             startIndex = indexOf(array, objectToFind, startIndex);
  2121.             if (startIndex == INDEX_NOT_FOUND) {
  2122.                 break;
  2123.             }
  2124.             bitSet.set(startIndex);
  2125.             ++startIndex;
  2126.         }
  2127.         return bitSet;
  2128.     }

  2129.     /**
  2130.      * Finds the indices of the given value in the array.
  2131.      *
  2132.      * <p>This method returns an empty BitSet for a {@code null} input array.</p>
  2133.      *
  2134.      * @param array  the array to search through for the object, may be {@code null}
  2135.      * @param valueToFind  the value to find
  2136.      * @return a BitSet of all the indices of the value within the array,
  2137.      *  an empty BitSet if not found or {@code null} array input
  2138.      * @since 3.10
  2139.      */
  2140.     public static BitSet indexesOf(final short[] array, final short valueToFind) {
  2141.         return indexesOf(array, valueToFind, 0);
  2142.     }

  2143.     /**
  2144.      * Finds the indices of the given value in the array starting at the given index.
  2145.      *
  2146.      * <p>This method returns an empty BitSet for a {@code null} input array.</p>
  2147.      *
  2148.      * <p>A negative startIndex is treated as zero. A startIndex larger than the array
  2149.      * length will return an empty BitSet.</p>
  2150.      *
  2151.      * @param array  the array to search through for the object, may be {@code null}
  2152.      * @param valueToFind  the value to find
  2153.      * @param startIndex  the index to start searching at
  2154.      * @return a BitSet of all the indices of the value within the array,
  2155.      *  an empty BitSet if not found or {@code null} array input
  2156.      * @since 3.10
  2157.      */
  2158.     public static BitSet indexesOf(final short[] array, final short valueToFind, int startIndex) {
  2159.         final BitSet bitSet = new BitSet();
  2160.         if (array == null) {
  2161.             return bitSet;
  2162.         }
  2163.         while (startIndex < array.length) {
  2164.             startIndex = indexOf(array, valueToFind, startIndex);
  2165.             if (startIndex == INDEX_NOT_FOUND) {
  2166.                 break;
  2167.             }
  2168.             bitSet.set(startIndex);
  2169.             ++startIndex;
  2170.         }
  2171.         return bitSet;
  2172.     }

  2173.     /**
  2174.      * Finds the index of the given value in the array.
  2175.      * <p>
  2176.      * This method returns {@link #INDEX_NOT_FOUND} ({@code -1}) for a {@code null} input array.
  2177.      * </p>
  2178.      *
  2179.      * @param array  the array to search through for the object, may be {@code null}
  2180.      * @param valueToFind  the value to find
  2181.      * @return the index of the value within the array,
  2182.      *  {@link #INDEX_NOT_FOUND} ({@code -1}) if not found or {@code null} array input
  2183.      */
  2184.     public static int indexOf(final boolean[] array, final boolean valueToFind) {
  2185.         return indexOf(array, valueToFind, 0);
  2186.     }

  2187.     /**
  2188.      * Finds the index of the given value in the array starting at the given index.
  2189.      * <p>
  2190.      * This method returns {@link #INDEX_NOT_FOUND} ({@code -1}) for a {@code null} input array.
  2191.      * </p>
  2192.      * <p>
  2193.      * A negative startIndex is treated as zero. A startIndex larger than the array
  2194.      * length will return {@link #INDEX_NOT_FOUND} ({@code -1}).
  2195.      * </p>
  2196.      *
  2197.      * @param array  the array to search through for the object, may be {@code null}
  2198.      * @param valueToFind  the value to find
  2199.      * @param startIndex  the index to start searching at
  2200.      * @return the index of the value within the array,
  2201.      *  {@link #INDEX_NOT_FOUND} ({@code -1}) if not found or {@code null}
  2202.      *  array input
  2203.      */
  2204.     public static int indexOf(final boolean[] array, final boolean valueToFind, final int startIndex) {
  2205.         if (isEmpty(array)) {
  2206.             return INDEX_NOT_FOUND;
  2207.         }
  2208.         for (int i = max0(startIndex); i < array.length; i++) {
  2209.             if (valueToFind == array[i]) {
  2210.                 return i;
  2211.             }
  2212.         }
  2213.         return INDEX_NOT_FOUND;
  2214.     }

  2215.     /**
  2216.      * Finds the index of the given value in the array.
  2217.      * <p>
  2218.      * This method returns {@link #INDEX_NOT_FOUND} ({@code -1}) for a {@code null} input array.
  2219.      * </p>
  2220.      *
  2221.      * @param array  the array to search through for the object, may be {@code null}
  2222.      * @param valueToFind  the value to find
  2223.      * @return the index of the value within the array,
  2224.      *  {@link #INDEX_NOT_FOUND} ({@code -1}) if not found or {@code null} array input
  2225.      */
  2226.     public static int indexOf(final byte[] array, final byte valueToFind) {
  2227.         return indexOf(array, valueToFind, 0);
  2228.     }

  2229.     /**
  2230.      * Finds the index of the given value in the array starting at the given index.
  2231.      * <p>
  2232.      * This method returns {@link #INDEX_NOT_FOUND} ({@code -1}) for a {@code null} input array.
  2233.      * </p>
  2234.      * <p>
  2235.      * A negative startIndex is treated as zero. A startIndex larger than the array
  2236.      * length will return {@link #INDEX_NOT_FOUND} ({@code -1}).
  2237.      * </p>
  2238.      *
  2239.      * @param array  the array to search through for the object, may be {@code null}
  2240.      * @param valueToFind  the value to find
  2241.      * @param startIndex  the index to start searching at
  2242.      * @return the index of the value within the array,
  2243.      *  {@link #INDEX_NOT_FOUND} ({@code -1}) if not found or {@code null} array input
  2244.      */
  2245.     public static int indexOf(final byte[] array, final byte valueToFind, final int startIndex) {
  2246.         if (array == null) {
  2247.             return INDEX_NOT_FOUND;
  2248.         }
  2249.         for (int i = max0(startIndex); i < array.length; i++) {
  2250.             if (valueToFind == array[i]) {
  2251.                 return i;
  2252.             }
  2253.         }
  2254.         return INDEX_NOT_FOUND;
  2255.     }

  2256.     /**
  2257.      * Finds the index of the given value in the array.
  2258.      * <p>
  2259.      * This method returns {@link #INDEX_NOT_FOUND} ({@code -1}) for a {@code null} input array.
  2260.      * </p>
  2261.      *
  2262.      * @param array  the array to search through for the object, may be {@code null}
  2263.      * @param valueToFind  the value to find
  2264.      * @return the index of the value within the array,
  2265.      *  {@link #INDEX_NOT_FOUND} ({@code -1}) if not found or {@code null} array input
  2266.      * @since 2.1
  2267.      */
  2268.     public static int indexOf(final char[] array, final char valueToFind) {
  2269.         return indexOf(array, valueToFind, 0);
  2270.     }

  2271.     /**
  2272.      * Finds the index of the given value in the array starting at the given index.
  2273.      * <p>
  2274.      * This method returns {@link #INDEX_NOT_FOUND} ({@code -1}) for a {@code null} input array.
  2275.      * </p>
  2276.      * <p>
  2277.      * A negative startIndex is treated as zero. A startIndex larger than the array
  2278.      * length will return {@link #INDEX_NOT_FOUND} ({@code -1}).
  2279.      * </p>
  2280.      *
  2281.      * @param array  the array to search through for the object, may be {@code null}
  2282.      * @param valueToFind  the value to find
  2283.      * @param startIndex  the index to start searching at
  2284.      * @return the index of the value within the array,
  2285.      *  {@link #INDEX_NOT_FOUND} ({@code -1}) if not found or {@code null} array input
  2286.      * @since 2.1
  2287.      */
  2288.     public static int indexOf(final char[] array, final char valueToFind, final int startIndex) {
  2289.         if (array == null) {
  2290.             return INDEX_NOT_FOUND;
  2291.         }
  2292.         for (int i = max0(startIndex); i < array.length; i++) {
  2293.             if (valueToFind == array[i]) {
  2294.                 return i;
  2295.             }
  2296.         }
  2297.         return INDEX_NOT_FOUND;
  2298.     }

  2299.     /**
  2300.      * Finds the index of the given value in the array.
  2301.      * <p>
  2302.      * This method returns {@link #INDEX_NOT_FOUND} ({@code -1}) for a {@code null} input array.
  2303.      * </p>
  2304.      *
  2305.      * @param array  the array to search through for the object, may be {@code null}
  2306.      * @param valueToFind  the value to find
  2307.      * @return the index of the value within the array,
  2308.      *  {@link #INDEX_NOT_FOUND} ({@code -1}) if not found or {@code null} array input
  2309.      */
  2310.     public static int indexOf(final double[] array, final double valueToFind) {
  2311.         return indexOf(array, valueToFind, 0);
  2312.     }

  2313.     /**
  2314.      * Finds the index of the given value within a given tolerance in the array.
  2315.      * This method will return the index of the first value which falls between the region
  2316.      * defined by valueToFind - tolerance and valueToFind + tolerance.
  2317.      * <p>
  2318.      * This method returns {@link #INDEX_NOT_FOUND} ({@code -1}) for a {@code null} input array.
  2319.      * </p>
  2320.      *
  2321.      * @param array  the array to search through for the object, may be {@code null}
  2322.      * @param valueToFind  the value to find
  2323.      * @param tolerance tolerance of the search
  2324.      * @return the index of the value within the array,
  2325.      *  {@link #INDEX_NOT_FOUND} ({@code -1}) if not found or {@code null} array input
  2326.      */
  2327.     public static int indexOf(final double[] array, final double valueToFind, final double tolerance) {
  2328.         return indexOf(array, valueToFind, 0, tolerance);
  2329.     }

  2330.     /**
  2331.      * Finds the index of the given value in the array starting at the given index.
  2332.      * <p>
  2333.      * This method returns {@link #INDEX_NOT_FOUND} ({@code -1}) for a {@code null} input array.
  2334.      * </p>
  2335.      * <p>
  2336.      * A negative startIndex is treated as zero. A startIndex larger than the array
  2337.      * length will return {@link #INDEX_NOT_FOUND} ({@code -1}).
  2338.      * </p>
  2339.      *
  2340.      * @param array  the array to search through for the object, may be {@code null}
  2341.      * @param valueToFind  the value to find
  2342.      * @param startIndex  the index to start searching at
  2343.      * @return the index of the value within the array,
  2344.      *  {@link #INDEX_NOT_FOUND} ({@code -1}) if not found or {@code null} array input
  2345.      */
  2346.     public static int indexOf(final double[] array, final double valueToFind, final int startIndex) {
  2347.         if (isEmpty(array)) {
  2348.             return INDEX_NOT_FOUND;
  2349.         }
  2350.         final boolean searchNaN = Double.isNaN(valueToFind);
  2351.         for (int i = max0(startIndex); i < array.length; i++) {
  2352.             final double element = array[i];
  2353.             if (valueToFind == element || searchNaN && Double.isNaN(element)) {
  2354.                 return i;
  2355.             }
  2356.         }
  2357.         return INDEX_NOT_FOUND;
  2358.     }

  2359.     /**
  2360.      * Finds the index of the given value in the array starting at the given index.
  2361.      * This method will return the index of the first value which falls between the region
  2362.      * defined by valueToFind - tolerance and valueToFind + tolerance.
  2363.      * <p>
  2364.      * This method returns {@link #INDEX_NOT_FOUND} ({@code -1}) for a {@code null} input array.
  2365.      * </p>
  2366.      * <p>
  2367.      * A negative startIndex is treated as zero. A startIndex larger than the array
  2368.      * length will return {@link #INDEX_NOT_FOUND} ({@code -1}).
  2369.      * </p>
  2370.      *
  2371.      * @param array  the array to search through for the object, may be {@code null}
  2372.      * @param valueToFind  the value to find
  2373.      * @param startIndex  the index to start searching at
  2374.      * @param tolerance tolerance of the search
  2375.      * @return the index of the value within the array,
  2376.      *  {@link #INDEX_NOT_FOUND} ({@code -1}) if not found or {@code null} array input
  2377.      */
  2378.     public static int indexOf(final double[] array, final double valueToFind, final int startIndex, final double tolerance) {
  2379.         if (isEmpty(array)) {
  2380.             return INDEX_NOT_FOUND;
  2381.         }
  2382.         final double min = valueToFind - tolerance;
  2383.         final double max = valueToFind + tolerance;
  2384.         for (int i = max0(startIndex); i < array.length; i++) {
  2385.             if (array[i] >= min && array[i] <= max) {
  2386.                 return i;
  2387.             }
  2388.         }
  2389.         return INDEX_NOT_FOUND;
  2390.     }

  2391.     /**
  2392.      * Finds the index of the given value in the array.
  2393.      * <p>
  2394.      * This method returns {@link #INDEX_NOT_FOUND} ({@code -1}) for a {@code null} input array.
  2395.      * </p>
  2396.      *
  2397.      * @param array  the array to search through for the object, may be {@code null}
  2398.      * @param valueToFind  the value to find
  2399.      * @return the index of the value within the array,
  2400.      *  {@link #INDEX_NOT_FOUND} ({@code -1}) if not found or {@code null} array input
  2401.      */
  2402.     public static int indexOf(final float[] array, final float valueToFind) {
  2403.         return indexOf(array, valueToFind, 0);
  2404.     }

  2405.     /**
  2406.      * Finds the index of the given value in the array starting at the given index.
  2407.      * <p>
  2408.      * This method returns {@link #INDEX_NOT_FOUND} ({@code -1}) for a {@code null} input array.
  2409.      * </p>
  2410.      * <p>
  2411.      * A negative startIndex is treated as zero. A startIndex larger than the array
  2412.      * length will return {@link #INDEX_NOT_FOUND} ({@code -1}).
  2413.      * </p>
  2414.      *
  2415.      * @param array  the array to search through for the object, may be {@code null}
  2416.      * @param valueToFind  the value to find
  2417.      * @param startIndex  the index to start searching at
  2418.      * @return the index of the value within the array,
  2419.      *  {@link #INDEX_NOT_FOUND} ({@code -1}) if not found or {@code null} array input
  2420.      */
  2421.     public static int indexOf(final float[] array, final float valueToFind, final int startIndex) {
  2422.         if (isEmpty(array)) {
  2423.             return INDEX_NOT_FOUND;
  2424.         }
  2425.         final boolean searchNaN = Float.isNaN(valueToFind);
  2426.         for (int i = max0(startIndex); i < array.length; i++) {
  2427.             final float element = array[i];
  2428.             if (valueToFind == element || searchNaN && Float.isNaN(element)) {
  2429.                 return i;
  2430.             }
  2431.         }
  2432.         return INDEX_NOT_FOUND;
  2433.     }

  2434.     /**
  2435.      * Finds the index of the given value in the array.
  2436.      * <p>
  2437.      * This method returns {@link #INDEX_NOT_FOUND} ({@code -1}) for a {@code null} input array.
  2438.      * </p>
  2439.      *
  2440.      * @param array  the array to search through for the object, may be {@code null}
  2441.      * @param valueToFind  the value to find
  2442.      * @return the index of the value within the array,
  2443.      *  {@link #INDEX_NOT_FOUND} ({@code -1}) if not found or {@code null} array input
  2444.      */
  2445.     public static int indexOf(final int[] array, final int valueToFind) {
  2446.         return indexOf(array, valueToFind, 0);
  2447.     }

  2448.     /**
  2449.      * Finds the index of the given value in the array starting at the given index.
  2450.      * <p>
  2451.      * This method returns {@link #INDEX_NOT_FOUND} ({@code -1}) for a {@code null} input array.
  2452.      * </p>
  2453.      * <p>
  2454.      * A negative startIndex is treated as zero. A startIndex larger than the array
  2455.      * length will return {@link #INDEX_NOT_FOUND} ({@code -1}).
  2456.      * </p>
  2457.      *
  2458.      * @param array  the array to search through for the object, may be {@code null}
  2459.      * @param valueToFind  the value to find
  2460.      * @param startIndex  the index to start searching at
  2461.      * @return the index of the value within the array,
  2462.      *  {@link #INDEX_NOT_FOUND} ({@code -1}) if not found or {@code null} array input
  2463.      */
  2464.     public static int indexOf(final int[] array, final int valueToFind, final int startIndex) {
  2465.         if (array == null) {
  2466.             return INDEX_NOT_FOUND;
  2467.         }
  2468.         for (int i = max0(startIndex); i < array.length; i++) {
  2469.             if (valueToFind == array[i]) {
  2470.                 return i;
  2471.             }
  2472.         }
  2473.         return INDEX_NOT_FOUND;
  2474.     }

  2475.     /**
  2476.      * Finds the index of the given value in the array.
  2477.      * <p>
  2478.      * This method returns {@link #INDEX_NOT_FOUND} ({@code -1}) for a {@code null} input array.
  2479.      * </p>
  2480.      *
  2481.      * @param array the array to search through for the object, may be {@code null}
  2482.      * @param valueToFind the value to find
  2483.      * @return the index of the value within the array, {@link #INDEX_NOT_FOUND} ({@code -1}) if not found or {@code null}
  2484.      *         array input
  2485.      */
  2486.     public static int indexOf(final long[] array, final long valueToFind) {
  2487.         return indexOf(array, valueToFind, 0);
  2488.     }

  2489.     /**
  2490.      * Finds the index of the given value in the array starting at the given index.
  2491.      * <p>
  2492.      * This method returns {@link #INDEX_NOT_FOUND} ({@code -1}) for a {@code null} input array.
  2493.      * </p>
  2494.      * <p>
  2495.      * A negative startIndex is treated as zero. A startIndex larger than the array
  2496.      * length will return {@link #INDEX_NOT_FOUND} ({@code -1}).
  2497.      * </p>
  2498.      *
  2499.      * @param array  the array to search through for the object, may be {@code null}
  2500.      * @param valueToFind  the value to find
  2501.      * @param startIndex  the index to start searching at
  2502.      * @return the index of the value within the array,
  2503.      *  {@link #INDEX_NOT_FOUND} ({@code -1}) if not found or {@code null} array input
  2504.      */
  2505.     public static int indexOf(final long[] array, final long valueToFind, final int startIndex) {
  2506.         if (array == null) {
  2507.             return INDEX_NOT_FOUND;
  2508.         }
  2509.         for (int i = max0(startIndex); i < array.length; i++) {
  2510.             if (valueToFind == array[i]) {
  2511.                 return i;
  2512.             }
  2513.         }
  2514.         return INDEX_NOT_FOUND;
  2515.     }

  2516.     /**
  2517.      * Finds the index of the given object in the array.
  2518.      * <p>
  2519.      * This method returns {@link #INDEX_NOT_FOUND} ({@code -1}) for a {@code null} input array.
  2520.      * </p>
  2521.      *
  2522.      * @param array  the array to search through for the object, may be {@code null}
  2523.      * @param objectToFind  the object to find, may be {@code null}
  2524.      * @return the index of the object within the array,
  2525.      *  {@link #INDEX_NOT_FOUND} ({@code -1}) if not found or {@code null} array input
  2526.      */
  2527.     public static int indexOf(final Object[] array, final Object objectToFind) {
  2528.         return indexOf(array, objectToFind, 0);
  2529.     }

  2530.     /**
  2531.      * Finds the index of the given object in the array starting at the given index.
  2532.      * <p>
  2533.      * This method returns {@link #INDEX_NOT_FOUND} ({@code -1}) for a {@code null} input array.
  2534.      * </p>
  2535.      * <p>
  2536.      * A negative startIndex is treated as zero. A startIndex larger than the array
  2537.      * length will return {@link #INDEX_NOT_FOUND} ({@code -1}).
  2538.      * </p>
  2539.      *
  2540.      * @param array  the array to search through for the object, may be {@code null}
  2541.      * @param objectToFind  the object to find, may be {@code null}
  2542.      * @param startIndex  the index to start searching at
  2543.      * @return the index of the object within the array starting at the index,
  2544.      *  {@link #INDEX_NOT_FOUND} ({@code -1}) if not found or {@code null} array input
  2545.      */
  2546.     public static int indexOf(final Object[] array, final Object objectToFind, int startIndex) {
  2547.         if (array == null) {
  2548.             return INDEX_NOT_FOUND;
  2549.         }
  2550.         startIndex = max0(startIndex);
  2551.         if (objectToFind == null) {
  2552.             for (int i = startIndex; i < array.length; i++) {
  2553.                 if (array[i] == null) {
  2554.                     return i;
  2555.                 }
  2556.             }
  2557.         } else {
  2558.             for (int i = startIndex; i < array.length; i++) {
  2559.                 if (objectToFind.equals(array[i])) {
  2560.                     return i;
  2561.                 }
  2562.             }
  2563.         }
  2564.         return INDEX_NOT_FOUND;
  2565.     }

  2566.     /**
  2567.      * Finds the index of the given value in the array.
  2568.      * <p>
  2569.      * This method returns {@link #INDEX_NOT_FOUND} ({@code -1}) for a {@code null} input array.
  2570.      * </p>
  2571.      *
  2572.      * @param array  the array to search through for the object, may be {@code null}
  2573.      * @param valueToFind  the value to find
  2574.      * @return the index of the value within the array,
  2575.      *  {@link #INDEX_NOT_FOUND} ({@code -1}) if not found or {@code null} array input
  2576.      */
  2577.     public static int indexOf(final short[] array, final short valueToFind) {
  2578.         return indexOf(array, valueToFind, 0);
  2579.     }

  2580.     /**
  2581.      * Finds the index of the given value in the array starting at the given index.
  2582.      * <p>
  2583.      * This method returns {@link #INDEX_NOT_FOUND} ({@code -1}) for a {@code null} input array.
  2584.      * </p>
  2585.      * <p>
  2586.      * A negative startIndex is treated as zero. A startIndex larger than the array
  2587.      * length will return {@link #INDEX_NOT_FOUND} ({@code -1}).
  2588.      * </p>
  2589.      *
  2590.      * @param array  the array to search through for the object, may be {@code null}
  2591.      * @param valueToFind  the value to find
  2592.      * @param startIndex  the index to start searching at
  2593.      * @return the index of the value within the array,
  2594.      *  {@link #INDEX_NOT_FOUND} ({@code -1}) if not found or {@code null} array input
  2595.      */
  2596.     public static int indexOf(final short[] array, final short valueToFind, final int startIndex) {
  2597.         if (array == null) {
  2598.             return INDEX_NOT_FOUND;
  2599.         }
  2600.         for (int i = max0(startIndex); i < array.length; i++) {
  2601.             if (valueToFind == array[i]) {
  2602.                 return i;
  2603.             }
  2604.         }
  2605.         return INDEX_NOT_FOUND;
  2606.     }

  2607.     /**
  2608.      * Inserts elements into an array at the given index (starting from zero).
  2609.      *
  2610.      * <p>When an array is returned, it is always a new array.</p>
  2611.      *
  2612.      * <pre>
  2613.      * ArrayUtils.insert(index, null, null)      = null
  2614.      * ArrayUtils.insert(index, array, null)     = cloned copy of 'array'
  2615.      * ArrayUtils.insert(index, null, values)    = null
  2616.      * </pre>
  2617.      *
  2618.      * @param index the position within {@code array} to insert the new values
  2619.      * @param array the array to insert the values into, may be {@code null}
  2620.      * @param values the new values to insert, may be {@code null}
  2621.      * @return The new array or {@code null} if the given array is {@code null}.
  2622.      * @throws IndexOutOfBoundsException if {@code array} is provided
  2623.      * and either {@code index < 0} or {@code index > array.length}
  2624.      * @since 3.6
  2625.      */
  2626.     public static boolean[] insert(final int index, final boolean[] array, final boolean... values) {
  2627.         if (array == null) {
  2628.             return null;
  2629.         }
  2630.         if (isEmpty(values)) {
  2631.             return clone(array);
  2632.         }
  2633.         if (index < 0 || index > array.length) {
  2634.             throw new IndexOutOfBoundsException("Index: " + index + ", Length: " + array.length);
  2635.         }
  2636.         final boolean[] result = new boolean[array.length + values.length];
  2637.         System.arraycopy(values, 0, result, index, values.length);
  2638.         if (index > 0) {
  2639.             System.arraycopy(array, 0, result, 0, index);
  2640.         }
  2641.         if (index < array.length) {
  2642.             System.arraycopy(array, index, result, index + values.length, array.length - index);
  2643.         }
  2644.         return result;
  2645.     }

  2646.     /**
  2647.      * Inserts elements into an array at the given index (starting from zero).
  2648.      *
  2649.      * <p>When an array is returned, it is always a new array.</p>
  2650.      *
  2651.      * <pre>
  2652.      * ArrayUtils.insert(index, null, null)      = null
  2653.      * ArrayUtils.insert(index, array, null)     = cloned copy of 'array'
  2654.      * ArrayUtils.insert(index, null, values)    = null
  2655.      * </pre>
  2656.      *
  2657.      * @param index the position within {@code array} to insert the new values
  2658.      * @param array the array to insert the values into, may be {@code null}
  2659.      * @param values the new values to insert, may be {@code null}
  2660.      * @return The new array or {@code null} if the given array is {@code null}.
  2661.      * @throws IndexOutOfBoundsException if {@code array} is provided
  2662.      * and either {@code index < 0} or {@code index > array.length}
  2663.      * @since 3.6
  2664.      */
  2665.     public static byte[] insert(final int index, final byte[] array, final byte... values) {
  2666.         if (array == null) {
  2667.             return null;
  2668.         }
  2669.         if (isEmpty(values)) {
  2670.             return clone(array);
  2671.         }
  2672.         if (index < 0 || index > array.length) {
  2673.             throw new IndexOutOfBoundsException("Index: " + index + ", Length: " + array.length);
  2674.         }
  2675.         final byte[] result = new byte[array.length + values.length];
  2676.         System.arraycopy(values, 0, result, index, values.length);
  2677.         if (index > 0) {
  2678.             System.arraycopy(array, 0, result, 0, index);
  2679.         }
  2680.         if (index < array.length) {
  2681.             System.arraycopy(array, index, result, index + values.length, array.length - index);
  2682.         }
  2683.         return result;
  2684.     }

  2685.     /**
  2686.      * Inserts elements into an array at the given index (starting from zero).
  2687.      *
  2688.      * <p>When an array is returned, it is always a new array.</p>
  2689.      *
  2690.      * <pre>
  2691.      * ArrayUtils.insert(index, null, null)      = null
  2692.      * ArrayUtils.insert(index, array, null)     = cloned copy of 'array'
  2693.      * ArrayUtils.insert(index, null, values)    = null
  2694.      * </pre>
  2695.      *
  2696.      * @param index the position within {@code array} to insert the new values
  2697.      * @param array the array to insert the values into, may be {@code null}
  2698.      * @param values the new values to insert, may be {@code null}
  2699.      * @return The new array or {@code null} if the given array is {@code null}.
  2700.      * @throws IndexOutOfBoundsException if {@code array} is provided
  2701.      * and either {@code index < 0} or {@code index > array.length}
  2702.      * @since 3.6
  2703.      */
  2704.     public static char[] insert(final int index, final char[] array, final char... values) {
  2705.         if (array == null) {
  2706.             return null;
  2707.         }
  2708.         if (isEmpty(values)) {
  2709.             return clone(array);
  2710.         }
  2711.         if (index < 0 || index > array.length) {
  2712.             throw new IndexOutOfBoundsException("Index: " + index + ", Length: " + array.length);
  2713.         }
  2714.         final char[] result = new char[array.length + values.length];
  2715.         System.arraycopy(values, 0, result, index, values.length);
  2716.         if (index > 0) {
  2717.             System.arraycopy(array, 0, result, 0, index);
  2718.         }
  2719.         if (index < array.length) {
  2720.             System.arraycopy(array, index, result, index + values.length, array.length - index);
  2721.         }
  2722.         return result;
  2723.     }

  2724.     /**
  2725.      * Inserts elements into an array at the given index (starting from zero).
  2726.      *
  2727.      * <p>When an array is returned, it is always a new array.</p>
  2728.      *
  2729.      * <pre>
  2730.      * ArrayUtils.insert(index, null, null)      = null
  2731.      * ArrayUtils.insert(index, array, null)     = cloned copy of 'array'
  2732.      * ArrayUtils.insert(index, null, values)    = null
  2733.      * </pre>
  2734.      *
  2735.      * @param index the position within {@code array} to insert the new values
  2736.      * @param array the array to insert the values into, may be {@code null}
  2737.      * @param values the new values to insert, may be {@code null}
  2738.      * @return The new array or {@code null} if the given array is {@code null}.
  2739.      * @throws IndexOutOfBoundsException if {@code array} is provided
  2740.      * and either {@code index < 0} or {@code index > array.length}
  2741.      * @since 3.6
  2742.      */
  2743.     public static double[] insert(final int index, final double[] array, final double... values) {
  2744.         if (array == null) {
  2745.             return null;
  2746.         }
  2747.         if (isEmpty(values)) {
  2748.             return clone(array);
  2749.         }
  2750.         if (index < 0 || index > array.length) {
  2751.             throw new IndexOutOfBoundsException("Index: " + index + ", Length: " + array.length);
  2752.         }
  2753.         final double[] result = new double[array.length + values.length];
  2754.         System.arraycopy(values, 0, result, index, values.length);
  2755.         if (index > 0) {
  2756.             System.arraycopy(array, 0, result, 0, index);
  2757.         }
  2758.         if (index < array.length) {
  2759.             System.arraycopy(array, index, result, index + values.length, array.length - index);
  2760.         }
  2761.         return result;
  2762.     }

  2763.     /**
  2764.      * Inserts elements into an array at the given index (starting from zero).
  2765.      *
  2766.      * <p>When an array is returned, it is always a new array.</p>
  2767.      *
  2768.      * <pre>
  2769.      * ArrayUtils.insert(index, null, null)      = null
  2770.      * ArrayUtils.insert(index, array, null)     = cloned copy of 'array'
  2771.      * ArrayUtils.insert(index, null, values)    = null
  2772.      * </pre>
  2773.      *
  2774.      * @param index the position within {@code array} to insert the new values
  2775.      * @param array the array to insert the values into, may be {@code null}
  2776.      * @param values the new values to insert, may be {@code null}
  2777.      * @return The new array or {@code null} if the given array is {@code null}.
  2778.      * @throws IndexOutOfBoundsException if {@code array} is provided
  2779.      * and either {@code index < 0} or {@code index > array.length}
  2780.      * @since 3.6
  2781.      */
  2782.     public static float[] insert(final int index, final float[] array, final float... values) {
  2783.         if (array == null) {
  2784.             return null;
  2785.         }
  2786.         if (isEmpty(values)) {
  2787.             return clone(array);
  2788.         }
  2789.         if (index < 0 || index > array.length) {
  2790.             throw new IndexOutOfBoundsException("Index: " + index + ", Length: " + array.length);
  2791.         }
  2792.         final float[] result = new float[array.length + values.length];
  2793.         System.arraycopy(values, 0, result, index, values.length);
  2794.         if (index > 0) {
  2795.             System.arraycopy(array, 0, result, 0, index);
  2796.         }
  2797.         if (index < array.length) {
  2798.             System.arraycopy(array, index, result, index + values.length, array.length - index);
  2799.         }
  2800.         return result;
  2801.     }

  2802.     /**
  2803.      * Inserts elements into an array at the given index (starting from zero).
  2804.      *
  2805.      * <p>When an array is returned, it is always a new array.</p>
  2806.      *
  2807.      * <pre>
  2808.      * ArrayUtils.insert(index, null, null)      = null
  2809.      * ArrayUtils.insert(index, array, null)     = cloned copy of 'array'
  2810.      * ArrayUtils.insert(index, null, values)    = null
  2811.      * </pre>
  2812.      *
  2813.      * @param index the position within {@code array} to insert the new values
  2814.      * @param array the array to insert the values into, may be {@code null}
  2815.      * @param values the new values to insert, may be {@code null}
  2816.      * @return The new array or {@code null} if the given array is {@code null}.
  2817.      * @throws IndexOutOfBoundsException if {@code array} is provided
  2818.      * and either {@code index < 0} or {@code index > array.length}
  2819.      * @since 3.6
  2820.      */
  2821.     public static int[] insert(final int index, final int[] array, final int... values) {
  2822.         if (array == null) {
  2823.             return null;
  2824.         }
  2825.         if (isEmpty(values)) {
  2826.             return clone(array);
  2827.         }
  2828.         if (index < 0 || index > array.length) {
  2829.             throw new IndexOutOfBoundsException("Index: " + index + ", Length: " + array.length);
  2830.         }
  2831.         final int[] result = new int[array.length + values.length];
  2832.         System.arraycopy(values, 0, result, index, values.length);
  2833.         if (index > 0) {
  2834.             System.arraycopy(array, 0, result, 0, index);
  2835.         }
  2836.         if (index < array.length) {
  2837.             System.arraycopy(array, index, result, index + values.length, array.length - index);
  2838.         }
  2839.         return result;
  2840.     }

  2841.     /**
  2842.      * Inserts elements into an array at the given index (starting from zero).
  2843.      *
  2844.      * <p>When an array is returned, it is always a new array.</p>
  2845.      *
  2846.      * <pre>
  2847.      * ArrayUtils.insert(index, null, null)      = null
  2848.      * ArrayUtils.insert(index, array, null)     = cloned copy of 'array'
  2849.      * ArrayUtils.insert(index, null, values)    = null
  2850.      * </pre>
  2851.      *
  2852.      * @param index the position within {@code array} to insert the new values
  2853.      * @param array the array to insert the values into, may be {@code null}
  2854.      * @param values the new values to insert, may be {@code null}
  2855.      * @return The new array or {@code null} if the given array is {@code null}.
  2856.      * @throws IndexOutOfBoundsException if {@code array} is provided
  2857.      * and either {@code index < 0} or {@code index > array.length}
  2858.      * @since 3.6
  2859.      */
  2860.     public static long[] insert(final int index, final long[] array, final long... values) {
  2861.         if (array == null) {
  2862.             return null;
  2863.         }
  2864.         if (isEmpty(values)) {
  2865.             return clone(array);
  2866.         }
  2867.         if (index < 0 || index > array.length) {
  2868.             throw new IndexOutOfBoundsException("Index: " + index + ", Length: " + array.length);
  2869.         }
  2870.         final long[] result = new long[array.length + values.length];
  2871.         System.arraycopy(values, 0, result, index, values.length);
  2872.         if (index > 0) {
  2873.             System.arraycopy(array, 0, result, 0, index);
  2874.         }
  2875.         if (index < array.length) {
  2876.             System.arraycopy(array, index, result, index + values.length, array.length - index);
  2877.         }
  2878.         return result;
  2879.     }

  2880.     /**
  2881.      * Inserts elements into an array at the given index (starting from zero).
  2882.      *
  2883.      * <p>When an array is returned, it is always a new array.</p>
  2884.      *
  2885.      * <pre>
  2886.      * ArrayUtils.insert(index, null, null)      = null
  2887.      * ArrayUtils.insert(index, array, null)     = cloned copy of 'array'
  2888.      * ArrayUtils.insert(index, null, values)    = null
  2889.      * </pre>
  2890.      *
  2891.      * @param index the position within {@code array} to insert the new values
  2892.      * @param array the array to insert the values into, may be {@code null}
  2893.      * @param values the new values to insert, may be {@code null}
  2894.      * @return The new array or {@code null} if the given array is {@code null}.
  2895.      * @throws IndexOutOfBoundsException if {@code array} is provided
  2896.      * and either {@code index < 0} or {@code index > array.length}
  2897.      * @since 3.6
  2898.      */
  2899.     public static short[] insert(final int index, final short[] array, final short... values) {
  2900.         if (array == null) {
  2901.             return null;
  2902.         }
  2903.         if (isEmpty(values)) {
  2904.             return clone(array);
  2905.         }
  2906.         if (index < 0 || index > array.length) {
  2907.             throw new IndexOutOfBoundsException("Index: " + index + ", Length: " + array.length);
  2908.         }
  2909.         final short[] result = new short[array.length + values.length];
  2910.         System.arraycopy(values, 0, result, index, values.length);
  2911.         if (index > 0) {
  2912.             System.arraycopy(array, 0, result, 0, index);
  2913.         }
  2914.         if (index < array.length) {
  2915.             System.arraycopy(array, index, result, index + values.length, array.length - index);
  2916.         }
  2917.         return result;
  2918.     }

  2919.     /**
  2920.      * Inserts elements into an array at the given index (starting from zero).
  2921.      *
  2922.      * <p>When an array is returned, it is always a new array.</p>
  2923.      *
  2924.      * <pre>
  2925.      * ArrayUtils.insert(index, null, null)      = null
  2926.      * ArrayUtils.insert(index, array, null)     = cloned copy of 'array'
  2927.      * ArrayUtils.insert(index, null, values)    = null
  2928.      * </pre>
  2929.      *
  2930.      * @param <T> The type of elements in {@code array} and {@code values}
  2931.      * @param index the position within {@code array} to insert the new values
  2932.      * @param array the array to insert the values into, may be {@code null}
  2933.      * @param values the new values to insert, may be {@code null}
  2934.      * @return The new array or {@code null} if the given array is {@code null}.
  2935.      * @throws IndexOutOfBoundsException if {@code array} is provided
  2936.      * and either {@code index < 0} or {@code index > array.length}
  2937.      * @since 3.6
  2938.      */
  2939.     @SafeVarargs
  2940.     public static <T> T[] insert(final int index, final T[] array, final T... values) {
  2941.         /*
  2942.          * Note on use of @SafeVarargs:
  2943.          *
  2944.          * By returning null when 'array' is null, we avoid returning the vararg
  2945.          * array to the caller. We also avoid relying on the type of the vararg
  2946.          * array, by inspecting the component type of 'array'.
  2947.          */
  2948.         if (array == null) {
  2949.             return null;
  2950.         }
  2951.         if (isEmpty(values)) {
  2952.             return clone(array);
  2953.         }
  2954.         if (index < 0 || index > array.length) {
  2955.             throw new IndexOutOfBoundsException("Index: " + index + ", Length: " + array.length);
  2956.         }
  2957.         final Class<T> type = getComponentType(array);
  2958.         final int length = array.length + values.length;
  2959.         final T[] result = newInstance(type, length);
  2960.         System.arraycopy(values, 0, result, index, values.length);
  2961.         if (index > 0) {
  2962.             System.arraycopy(array, 0, result, 0, index);
  2963.         }
  2964.         if (index < array.length) {
  2965.             System.arraycopy(array, index, result, index + values.length, array.length - index);
  2966.         }
  2967.         return result;
  2968.     }

  2969.     /**
  2970.      * Checks if an array is empty or {@code null}.
  2971.      *
  2972.      * @param array the array to test
  2973.      * @return {@code true} if the array is empty or {@code null}
  2974.      */
  2975.     private static boolean isArrayEmpty(final Object array) {
  2976.         return getLength(array) == 0;
  2977.     }

  2978.     /**
  2979.      * Returns whether a given array can safely be accessed at the given index.
  2980.      *
  2981.      * <pre>
  2982.      * ArrayUtils.isArrayIndexValid(null, 0)       = false
  2983.      * ArrayUtils.isArrayIndexValid([], 0)         = false
  2984.      * ArrayUtils.isArrayIndexValid(["a"], 0)      = true
  2985.      * </pre>
  2986.      *
  2987.      * @param <T> the component type of the array
  2988.      * @param array the array to inspect, may be null
  2989.      * @param index the index of the array to be inspected
  2990.      * @return Whether the given index is safely-accessible in the given array
  2991.      * @since 3.8
  2992.      */
  2993.     public static <T> boolean isArrayIndexValid(final T[] array, final int index) {
  2994.         return index >= 0 && getLength(array) > index;
  2995.     }

  2996.     /**
  2997.      * Checks if an array of primitive booleans is empty or {@code null}.
  2998.      *
  2999.      * @param array  the array to test
  3000.      * @return {@code true} if the array is empty or {@code null}
  3001.      * @since 2.1
  3002.      */
  3003.     public static boolean isEmpty(final boolean[] array) {
  3004.         return isArrayEmpty(array);
  3005.     }

  3006.     /**
  3007.      * Checks if an array of primitive bytes is empty or {@code null}.
  3008.      *
  3009.      * @param array  the array to test
  3010.      * @return {@code true} if the array is empty or {@code null}
  3011.      * @since 2.1
  3012.      */
  3013.     public static boolean isEmpty(final byte[] array) {
  3014.         return isArrayEmpty(array);
  3015.     }

  3016.     /**
  3017.      * Checks if an array of primitive chars is empty or {@code null}.
  3018.      *
  3019.      * @param array  the array to test
  3020.      * @return {@code true} if the array is empty or {@code null}
  3021.      * @since 2.1
  3022.      */
  3023.     public static boolean isEmpty(final char[] array) {
  3024.         return isArrayEmpty(array);
  3025.     }

  3026.     /**
  3027.      * Checks if an array of primitive doubles is empty or {@code null}.
  3028.      *
  3029.      * @param array  the array to test
  3030.      * @return {@code true} if the array is empty or {@code null}
  3031.      * @since 2.1
  3032.      */
  3033.     public static boolean isEmpty(final double[] array) {
  3034.         return isArrayEmpty(array);
  3035.     }

  3036.     /**
  3037.      * Checks if an array of primitive floats is empty or {@code null}.
  3038.      *
  3039.      * @param array  the array to test
  3040.      * @return {@code true} if the array is empty or {@code null}
  3041.      * @since 2.1
  3042.      */
  3043.     public static boolean isEmpty(final float[] array) {
  3044.         return isArrayEmpty(array);
  3045.     }

  3046.     /**
  3047.      * Checks if an array of primitive ints is empty or {@code null}.
  3048.      *
  3049.      * @param array  the array to test
  3050.      * @return {@code true} if the array is empty or {@code null}
  3051.      * @since 2.1
  3052.      */
  3053.     public static boolean isEmpty(final int[] array) {
  3054.         return isArrayEmpty(array);
  3055.     }

  3056.     /**
  3057.      * Checks if an array of primitive longs is empty or {@code null}.
  3058.      *
  3059.      * @param array  the array to test
  3060.      * @return {@code true} if the array is empty or {@code null}
  3061.      * @since 2.1
  3062.      */
  3063.     public static boolean isEmpty(final long[] array) {
  3064.         return isArrayEmpty(array);
  3065.     }

  3066.     /**
  3067.      * Checks if an array of Objects is empty or {@code null}.
  3068.      *
  3069.      * @param array  the array to test
  3070.      * @return {@code true} if the array is empty or {@code null}
  3071.      * @since 2.1
  3072.      */
  3073.     public static boolean isEmpty(final Object[] array) {
  3074.         return isArrayEmpty(array);
  3075.     }

  3076.     /**
  3077.      * Checks if an array of primitive shorts is empty or {@code null}.
  3078.      *
  3079.      * @param array  the array to test
  3080.      * @return {@code true} if the array is empty or {@code null}
  3081.      * @since 2.1
  3082.      */
  3083.     public static boolean isEmpty(final short[] array) {
  3084.         return isArrayEmpty(array);
  3085.     }

  3086.     /**
  3087.      * Compares two arrays, using equals(), handling multidimensional arrays
  3088.      * correctly.
  3089.      * <p>
  3090.      * Multi-dimensional primitive arrays are also handled correctly by this method.
  3091.      * </p>
  3092.      *
  3093.      * @param array1  the left-hand side array to compare, may be {@code null}
  3094.      * @param array2  the right-hand side array to compare, may be {@code null}
  3095.      * @return {@code true} if the arrays are equal
  3096.      * @deprecated this method has been replaced by {@code java.util.Objects.deepEquals(Object, Object)} and will be
  3097.      * removed from future releases.
  3098.      */
  3099.     @Deprecated
  3100.     public static boolean isEquals(final Object array1, final Object array2) {
  3101.         return new EqualsBuilder().append(array1, array2).isEquals();
  3102.     }

  3103.     /**
  3104.      * Checks if an array of primitive booleans is not empty and not {@code null}.
  3105.      *
  3106.      * @param array  the array to test
  3107.      * @return {@code true} if the array is not empty and not {@code null}
  3108.      * @since 2.5
  3109.      */
  3110.     public static boolean isNotEmpty(final boolean[] array) {
  3111.         return !isEmpty(array);
  3112.     }

  3113.     /**
  3114.      * Checks if an array of primitive bytes is not empty and not {@code null}.
  3115.      *
  3116.      * @param array  the array to test
  3117.      * @return {@code true} if the array is not empty and not {@code null}
  3118.      * @since 2.5
  3119.      */
  3120.     public static boolean isNotEmpty(final byte[] array) {
  3121.         return !isEmpty(array);
  3122.     }

  3123.     /**
  3124.      * Checks if an array of primitive chars is not empty and not {@code null}.
  3125.      *
  3126.      * @param array  the array to test
  3127.      * @return {@code true} if the array is not empty and not {@code null}
  3128.      * @since 2.5
  3129.      */
  3130.     public static boolean isNotEmpty(final char[] array) {
  3131.         return !isEmpty(array);
  3132.     }

  3133.     /**
  3134.      * Checks if an array of primitive doubles is not empty and not {@code null}.
  3135.      *
  3136.      * @param array  the array to test
  3137.      * @return {@code true} if the array is not empty and not {@code null}
  3138.      * @since 2.5
  3139.      */
  3140.     public static boolean isNotEmpty(final double[] array) {
  3141.         return !isEmpty(array);
  3142.     }

  3143.     /**
  3144.      * Checks if an array of primitive floats is not empty and not {@code null}.
  3145.      *
  3146.      * @param array  the array to test
  3147.      * @return {@code true} if the array is not empty and not {@code null}
  3148.      * @since 2.5
  3149.      */
  3150.     public static boolean isNotEmpty(final float[] array) {
  3151.         return !isEmpty(array);
  3152.     }

  3153.     /**
  3154.      * Checks if an array of primitive ints is not empty and not {@code null}.
  3155.      *
  3156.      * @param array  the array to test
  3157.      * @return {@code true} if the array is not empty and not {@code null}
  3158.      * @since 2.5
  3159.      */
  3160.     public static boolean isNotEmpty(final int[] array) {
  3161.         return !isEmpty(array);
  3162.     }

  3163.     /**
  3164.      * Checks if an array of primitive longs is not empty and not {@code null}.
  3165.      *
  3166.      * @param array  the array to test
  3167.      * @return {@code true} if the array is not empty and not {@code null}
  3168.      * @since 2.5
  3169.      */
  3170.     public static boolean isNotEmpty(final long[] array) {
  3171.         return !isEmpty(array);
  3172.     }

  3173.     /**
  3174.      * Checks if an array of primitive shorts is not empty and not {@code null}.
  3175.      *
  3176.      * @param array  the array to test
  3177.      * @return {@code true} if the array is not empty and not {@code null}
  3178.      * @since 2.5
  3179.      */
  3180.     public static boolean isNotEmpty(final short[] array) {
  3181.         return !isEmpty(array);
  3182.     }

  3183.     /**
  3184.      * Checks if an array of Objects is not empty and not {@code null}.
  3185.      *
  3186.      * @param <T> the component type of the array
  3187.      * @param array  the array to test
  3188.      * @return {@code true} if the array is not empty and not {@code null}
  3189.      * @since 2.5
  3190.      */
  3191.      public static <T> boolean isNotEmpty(final T[] array) {
  3192.          return !isEmpty(array);
  3193.      }

  3194.     /**
  3195.      * Checks whether two arrays are the same length, treating
  3196.      * {@code null} arrays as length {@code 0}.
  3197.      *
  3198.      * @param array1 the first array, may be {@code null}
  3199.      * @param array2 the second array, may be {@code null}
  3200.      * @return {@code true} if length of arrays matches, treating
  3201.      *  {@code null} as an empty array
  3202.      */
  3203.     public static boolean isSameLength(final boolean[] array1, final boolean[] array2) {
  3204.         return getLength(array1) == getLength(array2);
  3205.     }

  3206.     /**
  3207.      * Checks whether two arrays are the same length, treating
  3208.      * {@code null} arrays as length {@code 0}.
  3209.      *
  3210.      * @param array1 the first array, may be {@code null}
  3211.      * @param array2 the second array, may be {@code null}
  3212.      * @return {@code true} if length of arrays matches, treating
  3213.      *  {@code null} as an empty array
  3214.      */
  3215.     public static boolean isSameLength(final byte[] array1, final byte[] array2) {
  3216.         return getLength(array1) == getLength(array2);
  3217.     }

  3218.     /**
  3219.      * Checks whether two arrays are the same length, treating
  3220.      * {@code null} arrays as length {@code 0}.
  3221.      *
  3222.      * @param array1 the first array, may be {@code null}
  3223.      * @param array2 the second array, may be {@code null}
  3224.      * @return {@code true} if length of arrays matches, treating
  3225.      *  {@code null} as an empty array
  3226.      */
  3227.     public static boolean isSameLength(final char[] array1, final char[] array2) {
  3228.         return getLength(array1) == getLength(array2);
  3229.     }

  3230.     /**
  3231.      * Checks whether two arrays are the same length, treating
  3232.      * {@code null} arrays as length {@code 0}.
  3233.      *
  3234.      * @param array1 the first array, may be {@code null}
  3235.      * @param array2 the second array, may be {@code null}
  3236.      * @return {@code true} if length of arrays matches, treating
  3237.      *  {@code null} as an empty array
  3238.      */
  3239.     public static boolean isSameLength(final double[] array1, final double[] array2) {
  3240.         return getLength(array1) == getLength(array2);
  3241.     }

  3242.     /**
  3243.      * Checks whether two arrays are the same length, treating
  3244.      * {@code null} arrays as length {@code 0}.
  3245.      *
  3246.      * @param array1 the first array, may be {@code null}
  3247.      * @param array2 the second array, may be {@code null}
  3248.      * @return {@code true} if length of arrays matches, treating
  3249.      *  {@code null} as an empty array
  3250.      */
  3251.     public static boolean isSameLength(final float[] array1, final float[] array2) {
  3252.         return getLength(array1) == getLength(array2);
  3253.     }

  3254.     /**
  3255.      * Checks whether two arrays are the same length, treating
  3256.      * {@code null} arrays as length {@code 0}.
  3257.      *
  3258.      * @param array1 the first array, may be {@code null}
  3259.      * @param array2 the second array, may be {@code null}
  3260.      * @return {@code true} if length of arrays matches, treating
  3261.      *  {@code null} as an empty array
  3262.      */
  3263.     public static boolean isSameLength(final int[] array1, final int[] array2) {
  3264.         return getLength(array1) == getLength(array2);
  3265.     }

  3266.     /**
  3267.      * Checks whether two arrays are the same length, treating
  3268.      * {@code null} arrays as length {@code 0}.
  3269.      *
  3270.      * @param array1 the first array, may be {@code null}
  3271.      * @param array2 the second array, may be {@code null}
  3272.      * @return {@code true} if length of arrays matches, treating
  3273.      *  {@code null} as an empty array
  3274.      */
  3275.     public static boolean isSameLength(final long[] array1, final long[] array2) {
  3276.         return getLength(array1) == getLength(array2);
  3277.     }

  3278.     /**
  3279.      * Checks whether two arrays are the same length, treating
  3280.      * {@code null} arrays as length {@code 0}.
  3281.      * <p>
  3282.      * Any multi-dimensional aspects of the arrays are ignored.
  3283.      * </p>
  3284.      *
  3285.      * @param array1 the first array, may be {@code null}
  3286.      * @param array2 the second array, may be {@code null}
  3287.      * @return {@code true} if length of arrays matches, treating
  3288.      *  {@code null} as an empty array
  3289.      * @since 3.11
  3290.      */
  3291.     public static boolean isSameLength(final Object array1, final Object array2) {
  3292.         return getLength(array1) == getLength(array2);
  3293.     }

  3294.     /**
  3295.      * Checks whether two arrays are the same length, treating
  3296.      * {@code null} arrays as length {@code 0}.
  3297.      * <p>
  3298.      * Any multi-dimensional aspects of the arrays are ignored.
  3299.      * </p>
  3300.      *
  3301.      * @param array1 the first array, may be {@code null}
  3302.      * @param array2 the second array, may be {@code null}
  3303.      * @return {@code true} if length of arrays matches, treating
  3304.      *  {@code null} as an empty array
  3305.      */
  3306.     public static boolean isSameLength(final Object[] array1, final Object[] array2) {
  3307.         return getLength(array1) == getLength(array2);
  3308.     }

  3309.     /**
  3310.      * Checks whether two arrays are the same length, treating
  3311.      * {@code null} arrays as length {@code 0}.
  3312.      *
  3313.      * @param array1 the first array, may be {@code null}
  3314.      * @param array2 the second array, may be {@code null}
  3315.      * @return {@code true} if length of arrays matches, treating
  3316.      *  {@code null} as an empty array
  3317.      */
  3318.     public static boolean isSameLength(final short[] array1, final short[] array2) {
  3319.         return getLength(array1) == getLength(array2);
  3320.     }

  3321.     /**
  3322.      * Checks whether two arrays are the same type taking into account
  3323.      * multidimensional arrays.
  3324.      *
  3325.      * @param array1 the first array, must not be {@code null}
  3326.      * @param array2 the second array, must not be {@code null}
  3327.      * @return {@code true} if type of arrays matches
  3328.      * @throws IllegalArgumentException if either array is {@code null}
  3329.      */
  3330.     public static boolean isSameType(final Object array1, final Object array2) {
  3331.         if (array1 == null || array2 == null) {
  3332.             throw new IllegalArgumentException("The Array must not be null");
  3333.         }
  3334.         return array1.getClass().getName().equals(array2.getClass().getName());
  3335.     }

  3336.     /**
  3337.      * This method checks whether the provided array is sorted according to natural ordering
  3338.      * ({@code false} before {@code true}).
  3339.      *
  3340.      * @param array the array to check
  3341.      * @return whether the array is sorted according to natural ordering
  3342.      * @since 3.4
  3343.      */
  3344.     public static boolean isSorted(final boolean[] array) {
  3345.         if (getLength(array) < 2) {
  3346.             return true;
  3347.         }
  3348.         boolean previous = array[0];
  3349.         final int n = array.length;
  3350.         for (int i = 1; i < n; i++) {
  3351.             final boolean current = array[i];
  3352.             if (BooleanUtils.compare(previous, current) > 0) {
  3353.                 return false;
  3354.             }
  3355.             previous = current;
  3356.         }
  3357.         return true;
  3358.     }

  3359.     /**
  3360.      * Checks whether the provided array is sorted according to natural ordering.
  3361.      *
  3362.      * @param array the array to check
  3363.      * @return whether the array is sorted according to natural ordering
  3364.      * @since 3.4
  3365.      */
  3366.     public static boolean isSorted(final byte[] array) {
  3367.         if (getLength(array) < 2) {
  3368.             return true;
  3369.         }
  3370.         byte previous = array[0];
  3371.         final int n = array.length;
  3372.         for (int i = 1; i < n; i++) {
  3373.             final byte current = array[i];
  3374.             if (NumberUtils.compare(previous, current) > 0) {
  3375.                 return false;
  3376.             }
  3377.             previous = current;
  3378.         }
  3379.         return true;
  3380.     }

  3381.     /**
  3382.      * Checks whether the provided array is sorted according to natural ordering.
  3383.      *
  3384.      * @param array the array to check
  3385.      * @return whether the array is sorted according to natural ordering
  3386.      * @since 3.4
  3387.      */
  3388.     public static boolean isSorted(final char[] array) {
  3389.         if (getLength(array) < 2) {
  3390.             return true;
  3391.         }
  3392.         char previous = array[0];
  3393.         final int n = array.length;
  3394.         for (int i = 1; i < n; i++) {
  3395.             final char current = array[i];
  3396.             if (CharUtils.compare(previous, current) > 0) {
  3397.                 return false;
  3398.             }
  3399.             previous = current;
  3400.         }
  3401.         return true;
  3402.     }

  3403.     /**
  3404.      * This method checks whether the provided array is sorted according to natural ordering.
  3405.      *
  3406.      * @param array the array to check
  3407.      * @return whether the array is sorted according to natural ordering
  3408.      * @since 3.4
  3409.      */
  3410.     public static boolean isSorted(final double[] array) {
  3411.         if (getLength(array) < 2) {
  3412.             return true;
  3413.         }
  3414.         double previous = array[0];
  3415.         final int n = array.length;
  3416.         for (int i = 1; i < n; i++) {
  3417.             final double current = array[i];
  3418.             if (Double.compare(previous, current) > 0) {
  3419.                 return false;
  3420.             }
  3421.             previous = current;
  3422.         }
  3423.         return true;
  3424.     }

  3425.     /**
  3426.      * This method checks whether the provided array is sorted according to natural ordering.
  3427.      *
  3428.      * @param array the array to check
  3429.      * @return whether the array is sorted according to natural ordering
  3430.      * @since 3.4
  3431.      */
  3432.     public static boolean isSorted(final float[] array) {
  3433.         if (getLength(array) < 2) {
  3434.             return true;
  3435.         }
  3436.         float previous = array[0];
  3437.         final int n = array.length;
  3438.         for (int i = 1; i < n; i++) {
  3439.             final float current = array[i];
  3440.             if (Float.compare(previous, current) > 0) {
  3441.                 return false;
  3442.             }
  3443.             previous = current;
  3444.         }
  3445.         return true;
  3446.     }

  3447.     /**
  3448.      * This method checks whether the provided array is sorted according to natural ordering.
  3449.      *
  3450.      * @param array the array to check
  3451.      * @return whether the array is sorted according to natural ordering
  3452.      * @since 3.4
  3453.      */
  3454.     public static boolean isSorted(final int[] array) {
  3455.         if (getLength(array) < 2) {
  3456.             return true;
  3457.         }
  3458.         int previous = array[0];
  3459.         final int n = array.length;
  3460.         for (int i = 1; i < n; i++) {
  3461.             final int current = array[i];
  3462.             if (NumberUtils.compare(previous, current) > 0) {
  3463.                 return false;
  3464.             }
  3465.             previous = current;
  3466.         }
  3467.         return true;
  3468.     }

  3469.     /**
  3470.      * This method checks whether the provided array is sorted according to natural ordering.
  3471.      *
  3472.      * @param array the array to check
  3473.      * @return whether the array is sorted according to natural ordering
  3474.      * @since 3.4
  3475.      */
  3476.     public static boolean isSorted(final long[] array) {
  3477.         if (getLength(array) < 2) {
  3478.             return true;
  3479.         }
  3480.         long previous = array[0];
  3481.         final int n = array.length;
  3482.         for (int i = 1; i < n; i++) {
  3483.             final long current = array[i];
  3484.             if (NumberUtils.compare(previous, current) > 0) {
  3485.                 return false;
  3486.             }
  3487.             previous = current;
  3488.         }
  3489.         return true;
  3490.     }

  3491.     /**
  3492.      * This method checks whether the provided array is sorted according to natural ordering.
  3493.      *
  3494.      * @param array the array to check
  3495.      * @return whether the array is sorted according to natural ordering
  3496.      * @since 3.4
  3497.      */
  3498.     public static boolean isSorted(final short[] array) {
  3499.         if (getLength(array) < 2) {
  3500.             return true;
  3501.         }
  3502.         short previous = array[0];
  3503.         final int n = array.length;
  3504.         for (int i = 1; i < n; i++) {
  3505.             final short current = array[i];
  3506.             if (NumberUtils.compare(previous, current) > 0) {
  3507.                 return false;
  3508.             }
  3509.             previous = current;
  3510.         }
  3511.         return true;
  3512.     }

  3513.     /**
  3514.      * This method checks whether the provided array is sorted according to the class's
  3515.      * {@code compareTo} method.
  3516.      *
  3517.      * @param array the array to check
  3518.      * @param <T> the datatype of the array to check, it must implement {@link Comparable}
  3519.      * @return whether the array is sorted
  3520.      * @since 3.4
  3521.      */
  3522.     public static <T extends Comparable<? super T>> boolean isSorted(final T[] array) {
  3523.         return isSorted(array, Comparable::compareTo);
  3524.     }

  3525.     /**
  3526.      * This method checks whether the provided array is sorted according to the provided {@link Comparator}.
  3527.      *
  3528.      * @param array the array to check
  3529.      * @param comparator the {@link Comparator} to compare over
  3530.      * @param <T> the datatype of the array
  3531.      * @return whether the array is sorted
  3532.      * @throws NullPointerException if {@code comparator} is {@code null}
  3533.      * @since 3.4
  3534.      */
  3535.     public static <T> boolean isSorted(final T[] array, final Comparator<T> comparator) {
  3536.         Objects.requireNonNull(comparator, "comparator");
  3537.         if (getLength(array) < 2) {
  3538.             return true;
  3539.         }
  3540.         T previous = array[0];
  3541.         final int n = array.length;
  3542.         for (int i = 1; i < n; i++) {
  3543.             final T current = array[i];
  3544.             if (comparator.compare(previous, current) > 0) {
  3545.                 return false;
  3546.             }
  3547.             previous = current;
  3548.         }
  3549.         return true;
  3550.     }

  3551.     /**
  3552.      * Finds the last index of the given value within the array.
  3553.      * <p>
  3554.      * This method returns {@link #INDEX_NOT_FOUND} ({@code -1}) if
  3555.      * {@code null} array input.
  3556.      * </p>
  3557.      *
  3558.      * @param array  the array to traverse backwards looking for the object, may be {@code null}
  3559.      * @param valueToFind  the object to find
  3560.      * @return the last index of the value within the array,
  3561.      *  {@link #INDEX_NOT_FOUND} ({@code -1}) if not found or {@code null} array input
  3562.      */
  3563.     public static int lastIndexOf(final boolean[] array, final boolean valueToFind) {
  3564.         return lastIndexOf(array, valueToFind, Integer.MAX_VALUE);
  3565.     }

  3566.     /**
  3567.      * Finds the last index of the given value in the array starting at the given index.
  3568.      * <p>
  3569.      * This method returns {@link #INDEX_NOT_FOUND} ({@code -1}) for a {@code null} input array.
  3570.      * </p>
  3571.      * <p>
  3572.      * A negative startIndex will return {@link #INDEX_NOT_FOUND} ({@code -1}). A startIndex larger than
  3573.      * the array length will search from the end of the array.
  3574.      * </p>
  3575.      *
  3576.      * @param array  the array to traverse for looking for the object, may be {@code null}
  3577.      * @param valueToFind  the value to find
  3578.      * @param startIndex  the start index to traverse backwards from
  3579.      * @return the last index of the value within the array,
  3580.      *  {@link #INDEX_NOT_FOUND} ({@code -1}) if not found or {@code null} array input
  3581.      */
  3582.     public static int lastIndexOf(final boolean[] array, final boolean valueToFind, int startIndex) {
  3583.         if (isEmpty(array) || startIndex < 0) {
  3584.             return INDEX_NOT_FOUND;
  3585.         }
  3586.         if (startIndex >= array.length) {
  3587.             startIndex = array.length - 1;
  3588.         }
  3589.         for (int i = startIndex; i >= 0; i--) {
  3590.             if (valueToFind == array[i]) {
  3591.                 return i;
  3592.             }
  3593.         }
  3594.         return INDEX_NOT_FOUND;
  3595.     }

  3596.     /**
  3597.      * Finds the last index of the given value within the array.
  3598.      * <p>
  3599.      * This method returns {@link #INDEX_NOT_FOUND} ({@code -1}) for a {@code null} input array.
  3600.      * </p>
  3601.      *
  3602.      * @param array  the array to traverse backwards looking for the object, may be {@code null}
  3603.      * @param valueToFind  the object to find
  3604.      * @return the last index of the value within the array,
  3605.      *  {@link #INDEX_NOT_FOUND} ({@code -1}) if not found or {@code null} array input
  3606.      */
  3607.     public static int lastIndexOf(final byte[] array, final byte valueToFind) {
  3608.         return lastIndexOf(array, valueToFind, Integer.MAX_VALUE);
  3609.     }

  3610.     /**
  3611.      * Finds the last index of the given value in the array starting at the given index.
  3612.      * <p>
  3613.      * This method returns {@link #INDEX_NOT_FOUND} ({@code -1}) for a {@code null} input array.
  3614.      * </p>
  3615.      * <p>
  3616.      * A negative startIndex will return {@link #INDEX_NOT_FOUND} ({@code -1}). A startIndex larger than the
  3617.      * array length will search from the end of the array.
  3618.      * </p>
  3619.      *
  3620.      * @param array  the array to traverse for looking for the object, may be {@code null}
  3621.      * @param valueToFind  the value to find
  3622.      * @param startIndex  the start index to traverse backwards from
  3623.      * @return the last index of the value within the array,
  3624.      *  {@link #INDEX_NOT_FOUND} ({@code -1}) if not found or {@code null} array input
  3625.      */
  3626.     public static int lastIndexOf(final byte[] array, final byte valueToFind, int startIndex) {
  3627.         if (array == null || startIndex < 0) {
  3628.             return INDEX_NOT_FOUND;
  3629.         }
  3630.         if (startIndex >= array.length) {
  3631.             startIndex = array.length - 1;
  3632.         }
  3633.         for (int i = startIndex; i >= 0; i--) {
  3634.             if (valueToFind == array[i]) {
  3635.                 return i;
  3636.             }
  3637.         }
  3638.         return INDEX_NOT_FOUND;
  3639.     }

  3640.     /**
  3641.      * Finds the last index of the given value within the array.
  3642.      * <p>
  3643.      * This method returns {@link #INDEX_NOT_FOUND} ({@code -1}) for a {@code null} input array.
  3644.      * </p>
  3645.      *
  3646.      * @param array  the array to traverse backwards looking for the object, may be {@code null}
  3647.      * @param valueToFind  the object to find
  3648.      * @return the last index of the value within the array,
  3649.      *  {@link #INDEX_NOT_FOUND} ({@code -1}) if not found or {@code null} array input
  3650.      * @since 2.1
  3651.      */
  3652.     public static int lastIndexOf(final char[] array, final char valueToFind) {
  3653.         return lastIndexOf(array, valueToFind, Integer.MAX_VALUE);
  3654.     }

  3655.     /**
  3656.      * Finds the last index of the given value in the array starting at the given index.
  3657.      * <p>
  3658.      * This method returns {@link #INDEX_NOT_FOUND} ({@code -1}) for a {@code null} input array.
  3659.      * </p>
  3660.      * <p>
  3661.      * A negative startIndex will return {@link #INDEX_NOT_FOUND} ({@code -1}). A startIndex larger than the
  3662.      * array length will search from the end of the array.
  3663.      * </p>
  3664.      *
  3665.      * @param array  the array to traverse for looking for the object, may be {@code null}
  3666.      * @param valueToFind  the value to find
  3667.      * @param startIndex  the start index to traverse backwards from
  3668.      * @return the last index of the value within the array,
  3669.      *  {@link #INDEX_NOT_FOUND} ({@code -1}) if not found or {@code null} array input
  3670.      * @since 2.1
  3671.      */
  3672.     public static int lastIndexOf(final char[] array, final char valueToFind, int startIndex) {
  3673.         if (array == null || startIndex < 0) {
  3674.             return INDEX_NOT_FOUND;
  3675.         }
  3676.         if (startIndex >= array.length) {
  3677.             startIndex = array.length - 1;
  3678.         }
  3679.         for (int i = startIndex; i >= 0; i--) {
  3680.             if (valueToFind == array[i]) {
  3681.                 return i;
  3682.             }
  3683.         }
  3684.         return INDEX_NOT_FOUND;
  3685.     }

  3686.     /**
  3687.      * Finds the last index of the given value within the array.
  3688.      * <p>
  3689.      * This method returns {@link #INDEX_NOT_FOUND} ({@code -1}) for a {@code null} input array.
  3690.      * </p>
  3691.      *
  3692.      * @param array  the array to traverse backwards looking for the object, may be {@code null}
  3693.      * @param valueToFind  the object to find
  3694.      * @return the last index of the value within the array,
  3695.      *  {@link #INDEX_NOT_FOUND} ({@code -1}) if not found or {@code null} array input
  3696.      */
  3697.     public static int lastIndexOf(final double[] array, final double valueToFind) {
  3698.         return lastIndexOf(array, valueToFind, Integer.MAX_VALUE);
  3699.     }

  3700.     /**
  3701.      * Finds the last index of the given value within a given tolerance in the array.
  3702.      * This method will return the index of the last value which falls between the region
  3703.      * defined by valueToFind - tolerance and valueToFind + tolerance.
  3704.      * <p>
  3705.      * This method returns {@link #INDEX_NOT_FOUND} ({@code -1}) for a {@code null} input array.
  3706.      * </p>
  3707.      *
  3708.      * @param array  the array to search through for the object, may be {@code null}
  3709.      * @param valueToFind  the value to find
  3710.      * @param tolerance tolerance of the search
  3711.      * @return the index of the value within the array,
  3712.      *  {@link #INDEX_NOT_FOUND} ({@code -1}) if not found or {@code null} array input
  3713.      */
  3714.     public static int lastIndexOf(final double[] array, final double valueToFind, final double tolerance) {
  3715.         return lastIndexOf(array, valueToFind, Integer.MAX_VALUE, tolerance);
  3716.     }

  3717.     /**
  3718.      * Finds the last index of the given value in the array starting at the given index.
  3719.      * <p>
  3720.      * This method returns {@link #INDEX_NOT_FOUND} ({@code -1}) for a {@code null} input array.
  3721.      * </p>
  3722.      * <p>
  3723.      * A negative startIndex will return {@link #INDEX_NOT_FOUND} ({@code -1}). A startIndex larger than the
  3724.      * array length will search from the end of the array.
  3725.      * </p>
  3726.      *
  3727.      * @param array  the array to traverse for looking for the object, may be {@code null}
  3728.      * @param valueToFind  the value to find
  3729.      * @param startIndex  the start index to traverse backwards from
  3730.      * @return the last index of the value within the array,
  3731.      *  {@link #INDEX_NOT_FOUND} ({@code -1}) if not found or {@code null} array input
  3732.      */
  3733.     public static int lastIndexOf(final double[] array, final double valueToFind, int startIndex) {
  3734.         if (isEmpty(array) || startIndex < 0) {
  3735.             return INDEX_NOT_FOUND;
  3736.         }
  3737.         if (startIndex >= array.length) {
  3738.             startIndex = array.length - 1;
  3739.         }
  3740.         for (int i = startIndex; i >= 0; i--) {
  3741.             if (valueToFind == array[i]) {
  3742.                 return i;
  3743.             }
  3744.         }
  3745.         return INDEX_NOT_FOUND;
  3746.     }

  3747.     /**
  3748.      * Finds the last index of the given value in the array starting at the given index.
  3749.      * This method will return the index of the last value which falls between the region
  3750.      * defined by valueToFind - tolerance and valueToFind + tolerance.
  3751.      * <p>
  3752.      * This method returns {@link #INDEX_NOT_FOUND} ({@code -1}) for a {@code null} input array.
  3753.      * </p>
  3754.      * <p>
  3755.      * A negative startIndex will return {@link #INDEX_NOT_FOUND} ({@code -1}). A startIndex larger than the
  3756.      * array length will search from the end of the array.
  3757.      * </p>
  3758.      *
  3759.      * @param array  the array to traverse for looking for the object, may be {@code null}
  3760.      * @param valueToFind  the value to find
  3761.      * @param startIndex  the start index to traverse backwards from
  3762.      * @param tolerance  search for value within plus/minus this amount
  3763.      * @return the last index of the value within the array,
  3764.      *  {@link #INDEX_NOT_FOUND} ({@code -1}) if not found or {@code null} array input
  3765.      */
  3766.     public static int lastIndexOf(final double[] array, final double valueToFind, int startIndex, final double tolerance) {
  3767.         if (isEmpty(array) || startIndex < 0) {
  3768.             return INDEX_NOT_FOUND;
  3769.         }
  3770.         if (startIndex >= array.length) {
  3771.             startIndex = array.length - 1;
  3772.         }
  3773.         final double min = valueToFind - tolerance;
  3774.         final double max = valueToFind + tolerance;
  3775.         for (int i = startIndex; i >= 0; i--) {
  3776.             if (array[i] >= min && array[i] <= max) {
  3777.                 return i;
  3778.             }
  3779.         }
  3780.         return INDEX_NOT_FOUND;
  3781.     }

  3782.     /**
  3783.      * Finds the last index of the given value within the array.
  3784.      * <p>
  3785.      * This method returns {@link #INDEX_NOT_FOUND} ({@code -1}) for a {@code null} input array.
  3786.      * </p>
  3787.      *
  3788.      * @param array  the array to traverse backwards looking for the object, may be {@code null}
  3789.      * @param valueToFind  the object to find
  3790.      * @return the last index of the value within the array,
  3791.      *  {@link #INDEX_NOT_FOUND} ({@code -1}) if not found or {@code null} array input
  3792.      */
  3793.     public static int lastIndexOf(final float[] array, final float valueToFind) {
  3794.         return lastIndexOf(array, valueToFind, Integer.MAX_VALUE);
  3795.     }

  3796.     /**
  3797.      * Finds the last index of the given value in the array starting at the given index.
  3798.      * <p>
  3799.      * This method returns {@link #INDEX_NOT_FOUND} ({@code -1}) for a {@code null} input array.
  3800.      * </p>
  3801.      * <p>
  3802.      * A negative startIndex will return {@link #INDEX_NOT_FOUND} ({@code -1}). A startIndex larger than the
  3803.      * array length will search from the end of the array.
  3804.      * </p>
  3805.      *
  3806.      * @param array  the array to traverse for looking for the object, may be {@code null}
  3807.      * @param valueToFind  the value to find
  3808.      * @param startIndex  the start index to traverse backwards from
  3809.      * @return the last index of the value within the array,
  3810.      *  {@link #INDEX_NOT_FOUND} ({@code -1}) if not found or {@code null} array input
  3811.      */
  3812.     public static int lastIndexOf(final float[] array, final float valueToFind, int startIndex) {
  3813.         if (isEmpty(array) || startIndex < 0) {
  3814.             return INDEX_NOT_FOUND;
  3815.         }
  3816.         if (startIndex >= array.length) {
  3817.             startIndex = array.length - 1;
  3818.         }
  3819.         for (int i = startIndex; i >= 0; i--) {
  3820.             if (valueToFind == array[i]) {
  3821.                 return i;
  3822.             }
  3823.         }
  3824.         return INDEX_NOT_FOUND;
  3825.     }

  3826.     /**
  3827.      * Finds the last index of the given value within the array.
  3828.      * <p>
  3829.      * This method returns {@link #INDEX_NOT_FOUND} ({@code -1}) for a {@code null} input array.
  3830.      * </p>
  3831.      *
  3832.      * @param array  the array to traverse backwards looking for the object, may be {@code null}
  3833.      * @param valueToFind  the object to find
  3834.      * @return the last index of the value within the array,
  3835.      *  {@link #INDEX_NOT_FOUND} ({@code -1}) if not found or {@code null} array input
  3836.      */
  3837.     public static int lastIndexOf(final int[] array, final int valueToFind) {
  3838.         return lastIndexOf(array, valueToFind, Integer.MAX_VALUE);
  3839.     }

  3840.     /**
  3841.      * Finds the last index of the given value in the array starting at the given index.
  3842.      * <p>
  3843.      * This method returns {@link #INDEX_NOT_FOUND} ({@code -1}) for a {@code null} input array.
  3844.      * </p>
  3845.      * <p>
  3846.      * A negative startIndex will return {@link #INDEX_NOT_FOUND} ({@code -1}). A startIndex larger than the
  3847.      * array length will search from the end of the array.
  3848.      * </p>
  3849.      *
  3850.      * @param array  the array to traverse for looking for the object, may be {@code null}
  3851.      * @param valueToFind  the value to find
  3852.      * @param startIndex  the start index to traverse backwards from
  3853.      * @return the last index of the value within the array,
  3854.      *  {@link #INDEX_NOT_FOUND} ({@code -1}) if not found or {@code null} array input
  3855.      */
  3856.     public static int lastIndexOf(final int[] array, final int valueToFind, int startIndex) {
  3857.         if (array == null || startIndex < 0) {
  3858.             return INDEX_NOT_FOUND;
  3859.         }
  3860.         if (startIndex >= array.length) {
  3861.             startIndex = array.length - 1;
  3862.         }
  3863.         for (int i = startIndex; i >= 0; i--) {
  3864.             if (valueToFind == array[i]) {
  3865.                 return i;
  3866.             }
  3867.         }
  3868.         return INDEX_NOT_FOUND;
  3869.     }

  3870.     /**
  3871.      * Finds the last index of the given value within the array.
  3872.      * <p>
  3873.      * This method returns {@link #INDEX_NOT_FOUND} ({@code -1}) for a {@code null} input array.
  3874.      * </p>
  3875.      *
  3876.      * @param array  the array to traverse backwards looking for the object, may be {@code null}
  3877.      * @param valueToFind  the object to find
  3878.      * @return the last index of the value within the array,
  3879.      *  {@link #INDEX_NOT_FOUND} ({@code -1}) if not found or {@code null} array input
  3880.      */
  3881.     public static int lastIndexOf(final long[] array, final long valueToFind) {
  3882.         return lastIndexOf(array, valueToFind, Integer.MAX_VALUE);
  3883.     }

  3884.     /**
  3885.      * Finds the last index of the given value in the array starting at the given index.
  3886.      * <p>
  3887.      * This method returns {@link #INDEX_NOT_FOUND} ({@code -1}) for a {@code null} input array.
  3888.      * </p>
  3889.      * <p>
  3890.      * A negative startIndex will return {@link #INDEX_NOT_FOUND} ({@code -1}). A startIndex larger than the
  3891.      * array length will search from the end of the array.
  3892.      * </p>
  3893.      *
  3894.      * @param array  the array to traverse for looking for the object, may be {@code null}
  3895.      * @param valueToFind  the value to find
  3896.      * @param startIndex  the start index to traverse backwards from
  3897.      * @return the last index of the value within the array,
  3898.      *  {@link #INDEX_NOT_FOUND} ({@code -1}) if not found or {@code null} array input
  3899.      */
  3900.     public static int lastIndexOf(final long[] array, final long valueToFind, int startIndex) {
  3901.         if (array == null || startIndex < 0) {
  3902.             return INDEX_NOT_FOUND;
  3903.         }
  3904.         if (startIndex >= array.length) {
  3905.             startIndex = array.length - 1;
  3906.         }
  3907.         for (int i = startIndex; i >= 0; i--) {
  3908.             if (valueToFind == array[i]) {
  3909.                 return i;
  3910.             }
  3911.         }
  3912.         return INDEX_NOT_FOUND;
  3913.     }

  3914.     /**
  3915.      * Finds the last index of the given object within the array.
  3916.      * <p>
  3917.      * This method returns {@link #INDEX_NOT_FOUND} ({@code -1}) for a {@code null} input array.
  3918.      * </p>
  3919.      *
  3920.      * @param array  the array to traverse backwards looking for the object, may be {@code null}
  3921.      * @param objectToFind  the object to find, may be {@code null}
  3922.      * @return the last index of the object within the array,
  3923.      *  {@link #INDEX_NOT_FOUND} ({@code -1}) if not found or {@code null} array input
  3924.      */
  3925.     public static int lastIndexOf(final Object[] array, final Object objectToFind) {
  3926.         return lastIndexOf(array, objectToFind, Integer.MAX_VALUE);
  3927.     }

  3928.     /**
  3929.      * Finds the last index of the given object in the array starting at the given index.
  3930.      * <p>
  3931.      * This method returns {@link #INDEX_NOT_FOUND} ({@code -1}) for a {@code null} input array.
  3932.      * </p>
  3933.      * <p>
  3934.      * A negative startIndex will return {@link #INDEX_NOT_FOUND} ({@code -1}). A startIndex larger than
  3935.      * the array length will search from the end of the array.
  3936.      * </p>
  3937.      *
  3938.      * @param array  the array to traverse for looking for the object, may be {@code null}
  3939.      * @param objectToFind  the object to find, may be {@code null}
  3940.      * @param startIndex  the start index to traverse backwards from
  3941.      * @return the last index of the object within the array,
  3942.      *  {@link #INDEX_NOT_FOUND} ({@code -1}) if not found or {@code null} array input
  3943.      */
  3944.     public static int lastIndexOf(final Object[] array, final Object objectToFind, int startIndex) {
  3945.         if (array == null || startIndex < 0) {
  3946.             return INDEX_NOT_FOUND;
  3947.         }
  3948.         if (startIndex >= array.length) {
  3949.             startIndex = array.length - 1;
  3950.         }
  3951.         if (objectToFind == null) {
  3952.             for (int i = startIndex; i >= 0; i--) {
  3953.                 if (array[i] == null) {
  3954.                     return i;
  3955.                 }
  3956.             }
  3957.         } else if (array.getClass().getComponentType().isInstance(objectToFind)) {
  3958.             for (int i = startIndex; i >= 0; i--) {
  3959.                 if (objectToFind.equals(array[i])) {
  3960.                     return i;
  3961.                 }
  3962.             }
  3963.         }
  3964.         return INDEX_NOT_FOUND;
  3965.     }

  3966.     /**
  3967.      * Finds the last index of the given value within the array.
  3968.      * <p>
  3969.      * This method returns {@link #INDEX_NOT_FOUND} ({@code -1}) for a {@code null} input array.
  3970.      * </p>
  3971.      *
  3972.      * @param array  the array to traverse backwards looking for the object, may be {@code null}
  3973.      * @param valueToFind  the object to find
  3974.      * @return the last index of the value within the array,
  3975.      *  {@link #INDEX_NOT_FOUND} ({@code -1}) if not found or {@code null} array input
  3976.      */
  3977.     public static int lastIndexOf(final short[] array, final short valueToFind) {
  3978.         return lastIndexOf(array, valueToFind, Integer.MAX_VALUE);
  3979.     }

  3980.     /**
  3981.      * Finds the last index of the given value in the array starting at the given index.
  3982.      * <p>
  3983.      * This method returns {@link #INDEX_NOT_FOUND} ({@code -1}) for a {@code null} input array.
  3984.      * </p>
  3985.      * <p>
  3986.      * A negative startIndex will return {@link #INDEX_NOT_FOUND} ({@code -1}). A startIndex larger than the
  3987.      * array length will search from the end of the array.
  3988.      * </p>
  3989.      *
  3990.      * @param array  the array to traverse for looking for the object, may be {@code null}
  3991.      * @param valueToFind  the value to find
  3992.      * @param startIndex  the start index to traverse backwards from
  3993.      * @return the last index of the value within the array,
  3994.      *  {@link #INDEX_NOT_FOUND} ({@code -1}) if not found or {@code null} array input
  3995.      */
  3996.     public static int lastIndexOf(final short[] array, final short valueToFind, int startIndex) {
  3997.         if (array == null || startIndex < 0) {
  3998.             return INDEX_NOT_FOUND;
  3999.         }
  4000.         if (startIndex >= array.length) {
  4001.             startIndex = array.length - 1;
  4002.         }
  4003.         for (int i = startIndex; i >= 0; i--) {
  4004.             if (valueToFind == array[i]) {
  4005.                 return i;
  4006.             }
  4007.         }
  4008.         return INDEX_NOT_FOUND;
  4009.     }

  4010.     private static int max0(int other) {
  4011.         return Math.max(0, other);
  4012.     }

  4013.     /**
  4014.      * Delegates to {@link Array#newInstance(Class,int)} using generics.
  4015.      *
  4016.      * @param <T> The array type.
  4017.      * @param componentType The array class.
  4018.      * @param length the array length
  4019.      * @return The new array.
  4020.      * @throws NullPointerException if the specified {@code componentType} parameter is null.
  4021.      * @since 3.13.0
  4022.      */
  4023.     @SuppressWarnings("unchecked") // OK, because array and values are of type T
  4024.     public static <T> T[] newInstance(final Class<T> componentType, final int length) {
  4025.         return (T[]) Array.newInstance(componentType, length);
  4026.     }

  4027.     /**
  4028.      * Defensive programming technique to change a {@code null}
  4029.      * reference to an empty one.
  4030.      * <p>
  4031.      * This method returns a default array for a {@code null} input array.
  4032.      * </p>
  4033.      * <p>
  4034.      * As a memory optimizing technique an empty array passed in will be overridden with
  4035.      * the empty {@code public static} references in this class.
  4036.      * </p>
  4037.      *
  4038.      * @param <T> The array type.
  4039.      * @param array  the array to check for {@code null} or empty
  4040.      * @param defaultArray A default array, usually empty.
  4041.      * @return the same array, or defaultArray if {@code null} or empty input.
  4042.      * @since 3.15.0
  4043.      */
  4044.     public static  <T> T[] nullTo(final T[] array, final T[] defaultArray) {
  4045.         return isEmpty(array) ? defaultArray : array;
  4046.     }

  4047.     /**
  4048.      * Defensive programming technique to change a {@code null}
  4049.      * reference to an empty one.
  4050.      * <p>
  4051.      * This method returns an empty array for a {@code null} input array.
  4052.      * </p>
  4053.      * <p>
  4054.      * As a memory optimizing technique an empty array passed in will be overridden with
  4055.      * the empty {@code public static} references in this class.
  4056.      * </p>
  4057.      *
  4058.      * @param array  the array to check for {@code null} or empty
  4059.      * @return the same array, {@code public static} empty array if {@code null} or empty input
  4060.      * @since 2.5
  4061.      */
  4062.     public static boolean[] nullToEmpty(final boolean[] array) {
  4063.         return isEmpty(array) ? EMPTY_BOOLEAN_ARRAY : array;
  4064.     }

  4065.     /**
  4066.      * Defensive programming technique to change a {@code null}
  4067.      * reference to an empty one.
  4068.      * <p>
  4069.      * This method returns an empty array for a {@code null} input array.
  4070.      * </p>
  4071.      * <p>
  4072.      * As a memory optimizing technique an empty array passed in will be overridden with
  4073.      * the empty {@code public static} references in this class.
  4074.      * </p>
  4075.      *
  4076.      * @param array  the array to check for {@code null} or empty
  4077.      * @return the same array, {@code public static} empty array if {@code null} or empty input
  4078.      * @since 2.5
  4079.      */
  4080.     public static Boolean[] nullToEmpty(final Boolean[] array) {
  4081.         return nullTo(array, EMPTY_BOOLEAN_OBJECT_ARRAY);
  4082.     }

  4083.     /**
  4084.      * Defensive programming technique to change a {@code null}
  4085.      * reference to an empty one.
  4086.      * <p>
  4087.      * This method returns an empty array for a {@code null} input array.
  4088.      * </p>
  4089.      * <p>
  4090.      * As a memory optimizing technique an empty array passed in will be overridden with
  4091.      * the empty {@code public static} references in this class.
  4092.      * </p>
  4093.      *
  4094.      * @param array  the array to check for {@code null} or empty
  4095.      * @return the same array, {@code public static} empty array if {@code null} or empty input
  4096.      * @since 2.5
  4097.      */
  4098.     public static byte[] nullToEmpty(final byte[] array) {
  4099.         return isEmpty(array) ? EMPTY_BYTE_ARRAY : array;
  4100.     }

  4101.     /**
  4102.      * Defensive programming technique to change a {@code null}
  4103.      * reference to an empty one.
  4104.      * <p>
  4105.      * This method returns an empty array for a {@code null} input array.
  4106.      * </p>
  4107.      * <p>
  4108.      * As a memory optimizing technique an empty array passed in will be overridden with
  4109.      * the empty {@code public static} references in this class.
  4110.      * </p>
  4111.      *
  4112.      * @param array  the array to check for {@code null} or empty
  4113.      * @return the same array, {@code public static} empty array if {@code null} or empty input
  4114.      * @since 2.5
  4115.      */
  4116.     public static Byte[] nullToEmpty(final Byte[] array) {
  4117.         return nullTo(array, EMPTY_BYTE_OBJECT_ARRAY);
  4118.     }

  4119.     /**
  4120.      * Defensive programming technique to change a {@code null}
  4121.      * reference to an empty one.
  4122.      * <p>
  4123.      * This method returns an empty array for a {@code null} input array.
  4124.      * </p>
  4125.      * <p>
  4126.      * As a memory optimizing technique an empty array passed in will be overridden with
  4127.      * the empty {@code public static} references in this class.
  4128.      * </p>
  4129.      *
  4130.      * @param array  the array to check for {@code null} or empty
  4131.      * @return the same array, {@code public static} empty array if {@code null} or empty input
  4132.      * @since 2.5
  4133.      */
  4134.     public static char[] nullToEmpty(final char[] array) {
  4135.         return isEmpty(array) ? EMPTY_CHAR_ARRAY : array;
  4136.     }

  4137.     /**
  4138.      * Defensive programming technique to change a {@code null}
  4139.      * reference to an empty one.
  4140.      * <p>
  4141.      * This method returns an empty array for a {@code null} input array.
  4142.      * </p>
  4143.      * <p>
  4144.      * As a memory optimizing technique an empty array passed in will be overridden with
  4145.      * the empty {@code public static} references in this class.
  4146.      * </p>
  4147.      *
  4148.      * @param array  the array to check for {@code null} or empty
  4149.      * @return the same array, {@code public static} empty array if {@code null} or empty input
  4150.      * @since 2.5
  4151.      */
  4152.     public static Character[] nullToEmpty(final Character[] array) {
  4153.         return nullTo(array, EMPTY_CHARACTER_OBJECT_ARRAY);
  4154.     }

  4155.     /**
  4156.      * Defensive programming technique to change a {@code null}
  4157.      * reference to an empty one.
  4158.      * <p>
  4159.      * This method returns an empty array for a {@code null} input array.
  4160.      * </p>
  4161.      * <p>
  4162.      * As a memory optimizing technique an empty array passed in will be overridden with
  4163.      * the empty {@code public static} references in this class.
  4164.      * </p>
  4165.      *
  4166.      * @param array  the array to check for {@code null} or empty
  4167.      * @return the same array, {@code public static} empty array if {@code null} or empty input
  4168.      * @since 3.2
  4169.      */
  4170.     public static Class<?>[] nullToEmpty(final Class<?>[] array) {
  4171.         return nullTo(array, EMPTY_CLASS_ARRAY);
  4172.     }

  4173.     /**
  4174.      * Defensive programming technique to change a {@code null}
  4175.      * reference to an empty one.
  4176.      * <p>
  4177.      * This method returns an empty array for a {@code null} input array.
  4178.      * </p>
  4179.      * <p>
  4180.      * As a memory optimizing technique an empty array passed in will be overridden with
  4181.      * the empty {@code public static} references in this class.
  4182.      * </p>
  4183.      *
  4184.      * @param array  the array to check for {@code null} or empty
  4185.      * @return the same array, {@code public static} empty array if {@code null} or empty input
  4186.      * @since 2.5
  4187.      */
  4188.     public static double[] nullToEmpty(final double[] array) {
  4189.         return isEmpty(array) ? EMPTY_DOUBLE_ARRAY : array;
  4190.     }

  4191.     /**
  4192.      * Defensive programming technique to change a {@code null}
  4193.      * reference to an empty one.
  4194.      * <p>
  4195.      * This method returns an empty array for a {@code null} input array.
  4196.      * </p>
  4197.      * <p>
  4198.      * As a memory optimizing technique an empty array passed in will be overridden with
  4199.      * the empty {@code public static} references in this class.
  4200.      * </p>
  4201.      *
  4202.      * @param array  the array to check for {@code null} or empty
  4203.      * @return the same array, {@code public static} empty array if {@code null} or empty input
  4204.      * @since 2.5
  4205.      */
  4206.     public static Double[] nullToEmpty(final Double[] array) {
  4207.         return nullTo(array, EMPTY_DOUBLE_OBJECT_ARRAY);
  4208.     }

  4209.     /**
  4210.      * Defensive programming technique to change a {@code null}
  4211.      * reference to an empty one.
  4212.      * <p>
  4213.      * This method returns an empty array for a {@code null} input array.
  4214.      * </p>
  4215.      * <p>
  4216.      * As a memory optimizing technique an empty array passed in will be overridden with
  4217.      * the empty {@code public static} references in this class.
  4218.      * </p>
  4219.      *
  4220.      * @param array  the array to check for {@code null} or empty
  4221.      * @return the same array, {@code public static} empty array if {@code null} or empty input
  4222.      * @since 2.5
  4223.      */
  4224.     public static float[] nullToEmpty(final float[] array) {
  4225.         return isEmpty(array) ? EMPTY_FLOAT_ARRAY : array;
  4226.     }

  4227.     /**
  4228.      * Defensive programming technique to change a {@code null}
  4229.      * reference to an empty one.
  4230.      * <p>
  4231.      * This method returns an empty array for a {@code null} input array.
  4232.      * </p>
  4233.      * <p>
  4234.      * As a memory optimizing technique an empty array passed in will be overridden with
  4235.      * the empty {@code public static} references in this class.
  4236.      * </p>
  4237.      *
  4238.      * @param array  the array to check for {@code null} or empty
  4239.      * @return the same array, {@code public static} empty array if {@code null} or empty input
  4240.      * @since 2.5
  4241.      */
  4242.     public static Float[] nullToEmpty(final Float[] array) {
  4243.         return nullTo(array, EMPTY_FLOAT_OBJECT_ARRAY);
  4244.     }

  4245.     /**
  4246.      * Defensive programming technique to change a {@code null}
  4247.      * reference to an empty one.
  4248.      * <p>
  4249.      * This method returns an empty array for a {@code null} input array.
  4250.      * </p>
  4251.      * <p>
  4252.      * As a memory optimizing technique an empty array passed in will be overridden with
  4253.      * the empty {@code public static} references in this class.
  4254.      * </p>
  4255.      *
  4256.      * @param array  the array to check for {@code null} or empty
  4257.      * @return the same array, {@code public static} empty array if {@code null} or empty input
  4258.      * @since 2.5
  4259.      */
  4260.     public static int[] nullToEmpty(final int[] array) {
  4261.         return isEmpty(array) ? EMPTY_INT_ARRAY : array;
  4262.     }

  4263.     /**
  4264.      * Defensive programming technique to change a {@code null}
  4265.      * reference to an empty one.
  4266.      * <p>
  4267.      * This method returns an empty array for a {@code null} input array.
  4268.      * </p>
  4269.      * <p>
  4270.      * As a memory optimizing technique an empty array passed in will be overridden with
  4271.      * the empty {@code public static} references in this class.
  4272.      * </p>
  4273.      *
  4274.      * @param array  the array to check for {@code null} or empty
  4275.      * @return the same array, {@code public static} empty array if {@code null} or empty input
  4276.      * @since 2.5
  4277.      */
  4278.     public static Integer[] nullToEmpty(final Integer[] array) {
  4279.         return nullTo(array, EMPTY_INTEGER_OBJECT_ARRAY);
  4280.     }

  4281.     /**
  4282.      * Defensive programming technique to change a {@code null}
  4283.      * reference to an empty one.
  4284.      * <p>
  4285.      * This method returns an empty array for a {@code null} input array.
  4286.      * </p>
  4287.      * <p>
  4288.      * As a memory optimizing technique an empty array passed in will be overridden with
  4289.      * the empty {@code public static} references in this class.
  4290.      * </p>
  4291.      *
  4292.      * @param array  the array to check for {@code null} or empty
  4293.      * @return the same array, {@code public static} empty array if {@code null} or empty input
  4294.      * @since 2.5
  4295.      */
  4296.     public static long[] nullToEmpty(final long[] array) {
  4297.         return isEmpty(array) ? EMPTY_LONG_ARRAY : array;
  4298.     }

  4299.     /**
  4300.      * Defensive programming technique to change a {@code null}
  4301.      * reference to an empty one.
  4302.      * <p>
  4303.      * This method returns an empty array for a {@code null} input array.
  4304.      * </p>
  4305.      * <p>
  4306.      * As a memory optimizing technique an empty array passed in will be overridden with
  4307.      * the empty {@code public static} references in this class.
  4308.      * </p>
  4309.      *
  4310.      * @param array  the array to check for {@code null} or empty
  4311.      * @return the same array, {@code public static} empty array if {@code null} or empty input
  4312.      * @since 2.5
  4313.      */
  4314.     public static Long[] nullToEmpty(final Long[] array) {
  4315.         return nullTo(array, EMPTY_LONG_OBJECT_ARRAY);
  4316.     }

  4317.     /**
  4318.      * Defensive programming technique to change a {@code null}
  4319.      * reference to an empty one.
  4320.      * <p>
  4321.      * This method returns an empty array for a {@code null} input array.
  4322.      * </p>
  4323.      * <p>
  4324.      * As a memory optimizing technique an empty array passed in will be overridden with
  4325.      * the empty {@code public static} references in this class.
  4326.      * </p>
  4327.      *
  4328.      * @param array  the array to check for {@code null} or empty
  4329.      * @return the same array, {@code public static} empty array if {@code null} or empty input
  4330.      * @since 2.5
  4331.      */
  4332.     public static Object[] nullToEmpty(final Object[] array) {
  4333.         return nullTo(array, EMPTY_OBJECT_ARRAY);
  4334.     }

  4335.     /**
  4336.      * Defensive programming technique to change a {@code null}
  4337.      * reference to an empty one.
  4338.      * <p>
  4339.      * This method returns an empty array for a {@code null} input array.
  4340.      * </p>
  4341.      * <p>
  4342.      * As a memory optimizing technique an empty array passed in will be overridden with
  4343.      * the empty {@code public static} references in this class.
  4344.      * </p>
  4345.      *
  4346.      * @param array  the array to check for {@code null} or empty
  4347.      * @return the same array, {@code public static} empty array if {@code null} or empty input
  4348.      * @since 2.5
  4349.      */
  4350.     public static short[] nullToEmpty(final short[] array) {
  4351.         return isEmpty(array) ? EMPTY_SHORT_ARRAY : array;
  4352.     }

  4353.     /**
  4354.      * Defensive programming technique to change a {@code null}
  4355.      * reference to an empty one.
  4356.      * <p>
  4357.      * This method returns an empty array for a {@code null} input array.
  4358.      * </p>
  4359.      * <p>
  4360.      * As a memory optimizing technique an empty array passed in will be overridden with
  4361.      * the empty {@code public static} references in this class.
  4362.      * </p>
  4363.      *
  4364.      * @param array  the array to check for {@code null} or empty
  4365.      * @return the same array, {@code public static} empty array if {@code null} or empty input
  4366.      * @since 2.5
  4367.      */
  4368.     public static Short[] nullToEmpty(final Short[] array) {
  4369.         return nullTo(array, EMPTY_SHORT_OBJECT_ARRAY);
  4370.     }

  4371.     /**
  4372.      * Defensive programming technique to change a {@code null}
  4373.      * reference to an empty one.
  4374.      * <p>
  4375.      * This method returns an empty array for a {@code null} input array.
  4376.      * </p>
  4377.      * <p>
  4378.      * As a memory optimizing technique an empty array passed in will be overridden with
  4379.      * the empty {@code public static} references in this class.
  4380.      * </p>
  4381.      *
  4382.      * @param array  the array to check for {@code null} or empty
  4383.      * @return the same array, {@code public static} empty array if {@code null} or empty input
  4384.      * @since 2.5
  4385.      */
  4386.     public static String[] nullToEmpty(final String[] array) {
  4387.         return nullTo(array, EMPTY_STRING_ARRAY);
  4388.     }

  4389.     /**
  4390.      * Defensive programming technique to change a {@code null}
  4391.      * reference to an empty one.
  4392.      * <p>
  4393.      * This method returns an empty array for a {@code null} input array.
  4394.      * </p>
  4395.      *
  4396.      * @param array  the array to check for {@code null} or empty
  4397.      * @param type   the class representation of the desired array
  4398.      * @param <T>  the class type
  4399.      * @return the same array, {@code public static} empty array if {@code null}
  4400.      * @throws IllegalArgumentException if the type argument is null
  4401.      * @since 3.5
  4402.      */
  4403.     public static <T> T[] nullToEmpty(final T[] array, final Class<T[]> type) {
  4404.         if (type == null) {
  4405.             throw new IllegalArgumentException("The type must not be null");
  4406.         }
  4407.         if (array == null) {
  4408.             return type.cast(Array.newInstance(type.getComponentType(), 0));
  4409.         }
  4410.         return array;
  4411.     }

  4412.     private static ThreadLocalRandom random() {
  4413.         return ThreadLocalRandom.current();
  4414.     }

  4415.     /**
  4416.      * Removes the element at the specified position from the specified array.
  4417.      * All subsequent elements are shifted to the left (subtracts one from
  4418.      * their indices).
  4419.      * <p>
  4420.      * This method returns a new array with the same elements of the input
  4421.      * array except the element on the specified position. The component
  4422.      * type of the returned array is always the same as that of the input
  4423.      * array.
  4424.      * </p>
  4425.      * <p>
  4426.      * If the input array is {@code null}, an IndexOutOfBoundsException
  4427.      * will be thrown, because in that case no valid index can be specified.
  4428.      * </p>
  4429.      * <pre>
  4430.      * ArrayUtils.remove([true], 0)              = []
  4431.      * ArrayUtils.remove([true, false], 0)       = [false]
  4432.      * ArrayUtils.remove([true, false], 1)       = [true]
  4433.      * ArrayUtils.remove([true, true, false], 1) = [true, false]
  4434.      * </pre>
  4435.      *
  4436.      * @param array  the array to remove the element from, may not be {@code null}
  4437.      * @param index  the position of the element to be removed
  4438.      * @return A new array containing the existing elements except the element
  4439.      *         at the specified position.
  4440.      * @throws IndexOutOfBoundsException if the index is out of range
  4441.      * (index &lt; 0 || index &gt;= array.length), or if the array is {@code null}.
  4442.      * @since 2.1
  4443.      */
  4444.     public static boolean[] remove(final boolean[] array, final int index) {
  4445.         return (boolean[]) remove((Object) array, index);
  4446.     }

  4447.     /**
  4448.      * Removes the element at the specified position from the specified array.
  4449.      * All subsequent elements are shifted to the left (subtracts one from
  4450.      * their indices).
  4451.      * <p>
  4452.      * This method returns a new array with the same elements of the input
  4453.      * array except the element on the specified position. The component
  4454.      * type of the returned array is always the same as that of the input
  4455.      * array.
  4456.      * </p>
  4457.      * <p>
  4458.      * If the input array is {@code null}, an IndexOutOfBoundsException
  4459.      * will be thrown, because in that case no valid index can be specified.
  4460.      * </p>
  4461.      * <pre>
  4462.      * ArrayUtils.remove([1], 0)          = []
  4463.      * ArrayUtils.remove([1, 0], 0)       = [0]
  4464.      * ArrayUtils.remove([1, 0], 1)       = [1]
  4465.      * ArrayUtils.remove([1, 0, 1], 1)    = [1, 1]
  4466.      * </pre>
  4467.      *
  4468.      * @param array  the array to remove the element from, may not be {@code null}
  4469.      * @param index  the position of the element to be removed
  4470.      * @return A new array containing the existing elements except the element
  4471.      *         at the specified position.
  4472.      * @throws IndexOutOfBoundsException if the index is out of range
  4473.      * (index &lt; 0 || index &gt;= array.length), or if the array is {@code null}.
  4474.      * @since 2.1
  4475.      */
  4476.     public static byte[] remove(final byte[] array, final int index) {
  4477.         return (byte[]) remove((Object) array, index);
  4478.     }

  4479.     /**
  4480.      * Removes the element at the specified position from the specified array.
  4481.      * All subsequent elements are shifted to the left (subtracts one from
  4482.      * their indices).
  4483.      * <p>
  4484.      * This method returns a new array with the same elements of the input
  4485.      * array except the element on the specified position. The component
  4486.      * type of the returned array is always the same as that of the input
  4487.      * array.
  4488.      * </p>
  4489.      * <p>
  4490.      * If the input array is {@code null}, an IndexOutOfBoundsException
  4491.      * will be thrown, because in that case no valid index can be specified.
  4492.      * </p>
  4493.      * <pre>
  4494.      * ArrayUtils.remove(['a'], 0)           = []
  4495.      * ArrayUtils.remove(['a', 'b'], 0)      = ['b']
  4496.      * ArrayUtils.remove(['a', 'b'], 1)      = ['a']
  4497.      * ArrayUtils.remove(['a', 'b', 'c'], 1) = ['a', 'c']
  4498.      * </pre>
  4499.      *
  4500.      * @param array  the array to remove the element from, may not be {@code null}
  4501.      * @param index  the position of the element to be removed
  4502.      * @return A new array containing the existing elements except the element
  4503.      *         at the specified position.
  4504.      * @throws IndexOutOfBoundsException if the index is out of range
  4505.      * (index &lt; 0 || index &gt;= array.length), or if the array is {@code null}.
  4506.      * @since 2.1
  4507.      */
  4508.     public static char[] remove(final char[] array, final int index) {
  4509.         return (char[]) remove((Object) array, index);
  4510.     }

  4511.     /**
  4512.      * Removes the element at the specified position from the specified array.
  4513.      * All subsequent elements are shifted to the left (subtracts one from
  4514.      * their indices).
  4515.      * <p>
  4516.      * This method returns a new array with the same elements of the input
  4517.      * array except the element on the specified position. The component
  4518.      * type of the returned array is always the same as that of the input
  4519.      * array.
  4520.      * </p>
  4521.      * <p>
  4522.      * If the input array is {@code null}, an IndexOutOfBoundsException
  4523.      * will be thrown, because in that case no valid index can be specified.
  4524.      * </p>
  4525.      * <pre>
  4526.      * ArrayUtils.remove([1.1], 0)           = []
  4527.      * ArrayUtils.remove([2.5, 6.0], 0)      = [6.0]
  4528.      * ArrayUtils.remove([2.5, 6.0], 1)      = [2.5]
  4529.      * ArrayUtils.remove([2.5, 6.0, 3.8], 1) = [2.5, 3.8]
  4530.      * </pre>
  4531.      *
  4532.      * @param array  the array to remove the element from, may not be {@code null}
  4533.      * @param index  the position of the element to be removed
  4534.      * @return A new array containing the existing elements except the element
  4535.      *         at the specified position.
  4536.      * @throws IndexOutOfBoundsException if the index is out of range
  4537.      * (index &lt; 0 || index &gt;= array.length), or if the array is {@code null}.
  4538.      * @since 2.1
  4539.      */
  4540.     public static double[] remove(final double[] array, final int index) {
  4541.         return (double[]) remove((Object) array, index);
  4542.     }

  4543.     /**
  4544.      * Removes the element at the specified position from the specified array.
  4545.      * All subsequent elements are shifted to the left (subtracts one from
  4546.      * their indices).
  4547.      * <p>
  4548.      * This method returns a new array with the same elements of the input
  4549.      * array except the element on the specified position. The component
  4550.      * type of the returned array is always the same as that of the input
  4551.      * array.
  4552.      * </p>
  4553.      * <p>
  4554.      * If the input array is {@code null}, an IndexOutOfBoundsException
  4555.      * will be thrown, because in that case no valid index can be specified.
  4556.      * </p>
  4557.      * <pre>
  4558.      * ArrayUtils.remove([1.1], 0)           = []
  4559.      * ArrayUtils.remove([2.5, 6.0], 0)      = [6.0]
  4560.      * ArrayUtils.remove([2.5, 6.0], 1)      = [2.5]
  4561.      * ArrayUtils.remove([2.5, 6.0, 3.8], 1) = [2.5, 3.8]
  4562.      * </pre>
  4563.      *
  4564.      * @param array  the array to remove the element from, may not be {@code null}
  4565.      * @param index  the position of the element to be removed
  4566.      * @return A new array containing the existing elements except the element
  4567.      *         at the specified position.
  4568.      * @throws IndexOutOfBoundsException if the index is out of range
  4569.      * (index &lt; 0 || index &gt;= array.length), or if the array is {@code null}.
  4570.      * @since 2.1
  4571.      */
  4572.     public static float[] remove(final float[] array, final int index) {
  4573.         return (float[]) remove((Object) array, index);
  4574.     }

  4575.     /**
  4576.      * Removes the element at the specified position from the specified array.
  4577.      * All subsequent elements are shifted to the left (subtracts one from
  4578.      * their indices).
  4579.      * <p>
  4580.      * This method returns a new array with the same elements of the input
  4581.      * array except the element on the specified position. The component
  4582.      * type of the returned array is always the same as that of the input
  4583.      * array.
  4584.      * </p>
  4585.      * <p>
  4586.      * If the input array is {@code null}, an IndexOutOfBoundsException
  4587.      * will be thrown, because in that case no valid index can be specified.
  4588.      * </p>
  4589.      * <pre>
  4590.      * ArrayUtils.remove([1], 0)         = []
  4591.      * ArrayUtils.remove([2, 6], 0)      = [6]
  4592.      * ArrayUtils.remove([2, 6], 1)      = [2]
  4593.      * ArrayUtils.remove([2, 6, 3], 1)   = [2, 3]
  4594.      * </pre>
  4595.      *
  4596.      * @param array  the array to remove the element from, may not be {@code null}
  4597.      * @param index  the position of the element to be removed
  4598.      * @return A new array containing the existing elements except the element
  4599.      *         at the specified position.
  4600.      * @throws IndexOutOfBoundsException if the index is out of range
  4601.      * (index &lt; 0 || index &gt;= array.length), or if the array is {@code null}.
  4602.      * @since 2.1
  4603.      */
  4604.     public static int[] remove(final int[] array, final int index) {
  4605.         return (int[]) remove((Object) array, index);
  4606.     }

  4607.     /**
  4608.      * Removes the element at the specified position from the specified array.
  4609.      * All subsequent elements are shifted to the left (subtracts one from
  4610.      * their indices).
  4611.      * <p>
  4612.      * This method returns a new array with the same elements of the input
  4613.      * array except the element on the specified position. The component
  4614.      * type of the returned array is always the same as that of the input
  4615.      * array.
  4616.      * </p>
  4617.      * <p>
  4618.      * If the input array is {@code null}, an IndexOutOfBoundsException
  4619.      * will be thrown, because in that case no valid index can be specified.
  4620.      * </p>
  4621.      * <pre>
  4622.      * ArrayUtils.remove([1], 0)         = []
  4623.      * ArrayUtils.remove([2, 6], 0)      = [6]
  4624.      * ArrayUtils.remove([2, 6], 1)      = [2]
  4625.      * ArrayUtils.remove([2, 6, 3], 1)   = [2, 3]
  4626.      * </pre>
  4627.      *
  4628.      * @param array  the array to remove the element from, may not be {@code null}
  4629.      * @param index  the position of the element to be removed
  4630.      * @return A new array containing the existing elements except the element
  4631.      *         at the specified position.
  4632.      * @throws IndexOutOfBoundsException if the index is out of range
  4633.      * (index &lt; 0 || index &gt;= array.length), or if the array is {@code null}.
  4634.      * @since 2.1
  4635.      */
  4636.     public static long[] remove(final long[] array, final int index) {
  4637.         return (long[]) remove((Object) array, index);
  4638.     }

  4639.     /**
  4640.      * Removes the element at the specified position from the specified array.
  4641.      * All subsequent elements are shifted to the left (subtracts one from
  4642.      * their indices).
  4643.      * <p>
  4644.      * This method returns a new array with the same elements of the input
  4645.      * array except the element on the specified position. The component
  4646.      * type of the returned array is always the same as that of the input
  4647.      * array.
  4648.      * </p>
  4649.      * <p>
  4650.      * If the input array is {@code null}, an IndexOutOfBoundsException
  4651.      * will be thrown, because in that case no valid index can be specified.
  4652.      * </p>
  4653.      *
  4654.      * @param array  the array to remove the element from, may not be {@code null}
  4655.      * @param index  the position of the element to be removed
  4656.      * @return A new array containing the existing elements except the element
  4657.      *         at the specified position.
  4658.      * @throws IndexOutOfBoundsException if the index is out of range
  4659.      * (index &lt; 0 || index &gt;= array.length), or if the array is {@code null}.
  4660.      * @since 2.1
  4661.      */
  4662.     private static Object remove(final Object array, final int index) {
  4663.         final int length = getLength(array);
  4664.         if (index < 0 || index >= length) {
  4665.             throw new IndexOutOfBoundsException("Index: " + index + ", Length: " + length);
  4666.         }
  4667.         final Object result = Array.newInstance(array.getClass().getComponentType(), length - 1);
  4668.         System.arraycopy(array, 0, result, 0, index);
  4669.         if (index < length - 1) {
  4670.             System.arraycopy(array, index + 1, result, index, length - index - 1);
  4671.         }
  4672.         return result;
  4673.     }

  4674.     /**
  4675.      * Removes the element at the specified position from the specified array.
  4676.      * All subsequent elements are shifted to the left (subtracts one from
  4677.      * their indices).
  4678.      * <p>
  4679.      * This method returns a new array with the same elements of the input
  4680.      * array except the element on the specified position. The component
  4681.      * type of the returned array is always the same as that of the input
  4682.      * array.
  4683.      * </p>
  4684.      * <p>
  4685.      * If the input array is {@code null}, an IndexOutOfBoundsException
  4686.      * will be thrown, because in that case no valid index can be specified.
  4687.      * </p>
  4688.      * <pre>
  4689.      * ArrayUtils.remove([1], 0)         = []
  4690.      * ArrayUtils.remove([2, 6], 0)      = [6]
  4691.      * ArrayUtils.remove([2, 6], 1)      = [2]
  4692.      * ArrayUtils.remove([2, 6, 3], 1)   = [2, 3]
  4693.      * </pre>
  4694.      *
  4695.      * @param array  the array to remove the element from, may not be {@code null}
  4696.      * @param index  the position of the element to be removed
  4697.      * @return A new array containing the existing elements except the element
  4698.      *         at the specified position.
  4699.      * @throws IndexOutOfBoundsException if the index is out of range
  4700.      * (index &lt; 0 || index &gt;= array.length), or if the array is {@code null}.
  4701.      * @since 2.1
  4702.      */
  4703.     public static short[] remove(final short[] array, final int index) {
  4704.         return (short[]) remove((Object) array, index);
  4705.     }

  4706.     /**
  4707.      * Removes the element at the specified position from the specified array.
  4708.      * All subsequent elements are shifted to the left (subtracts one from
  4709.      * their indices).
  4710.      * <p>
  4711.      * This method returns a new array with the same elements of the input
  4712.      * array except the element on the specified position. The component
  4713.      * type of the returned array is always the same as that of the input
  4714.      * array.
  4715.      * </p>
  4716.      * <p>
  4717.      * If the input array is {@code null}, an IndexOutOfBoundsException
  4718.      * will be thrown, because in that case no valid index can be specified.
  4719.      * </p>
  4720.      * <pre>
  4721.      * ArrayUtils.remove(["a"], 0)           = []
  4722.      * ArrayUtils.remove(["a", "b"], 0)      = ["b"]
  4723.      * ArrayUtils.remove(["a", "b"], 1)      = ["a"]
  4724.      * ArrayUtils.remove(["a", "b", "c"], 1) = ["a", "c"]
  4725.      * </pre>
  4726.      *
  4727.      * @param <T> the component type of the array
  4728.      * @param array  the array to remove the element from, may not be {@code null}
  4729.      * @param index  the position of the element to be removed
  4730.      * @return A new array containing the existing elements except the element
  4731.      *         at the specified position.
  4732.      * @throws IndexOutOfBoundsException if the index is out of range
  4733.      * (index &lt; 0 || index &gt;= array.length), or if the array is {@code null}.
  4734.      * @since 2.1
  4735.      */
  4736.     @SuppressWarnings("unchecked") // remove() always creates an array of the same type as its input
  4737.     public static <T> T[] remove(final T[] array, final int index) {
  4738.         return (T[]) remove((Object) array, index);
  4739.     }

  4740.     /**
  4741.      * Removes the elements at the specified positions from the specified array.
  4742.      * All remaining elements are shifted to the left.
  4743.      * <p>
  4744.      * This method returns a new array with the same elements of the input
  4745.      * array except those at the specified positions. The component
  4746.      * type of the returned array is always the same as that of the input
  4747.      * array.
  4748.      * </p>
  4749.      * <p>
  4750.      * If the input array is {@code null}, an IndexOutOfBoundsException
  4751.      * will be thrown, because in that case no valid index can be specified.
  4752.      * </p>
  4753.      * <pre>
  4754.      * ArrayUtils.removeAll([true, false, true], 0, 2) = [false]
  4755.      * ArrayUtils.removeAll([true, false, true], 1, 2) = [true]
  4756.      * </pre>
  4757.      *
  4758.      * @param array   the array to remove the element from, may not be {@code null}
  4759.      * @param indices the positions of the elements to be removed
  4760.      * @return A new array containing the existing elements except those
  4761.      *         at the specified positions.
  4762.      * @throws IndexOutOfBoundsException if any index is out of range
  4763.      * (index &lt; 0 || index &gt;= array.length), or if the array is {@code null}.
  4764.      * @since 3.0.1
  4765.      */
  4766.     public static boolean[] removeAll(final boolean[] array, final int... indices) {
  4767.         return (boolean[]) removeAll((Object) array, indices);
  4768.     }

  4769.     /**
  4770.      * Removes the elements at the specified positions from the specified array.
  4771.      * All remaining elements are shifted to the left.
  4772.      * <p>
  4773.      * This method returns a new array with the same elements of the input
  4774.      * array except those at the specified positions. The component
  4775.      * type of the returned array is always the same as that of the input
  4776.      * array.
  4777.      * </p>
  4778.      * <p>
  4779.      * If the input array is {@code null}, an IndexOutOfBoundsException
  4780.      * will be thrown, because in that case no valid index can be specified.
  4781.      * </p>
  4782.      * <pre>
  4783.      * ArrayUtils.removeAll([1], 0)             = []
  4784.      * ArrayUtils.removeAll([2, 6], 0)          = [6]
  4785.      * ArrayUtils.removeAll([2, 6], 0, 1)       = []
  4786.      * ArrayUtils.removeAll([2, 6, 3], 1, 2)    = [2]
  4787.      * ArrayUtils.removeAll([2, 6, 3], 0, 2)    = [6]
  4788.      * ArrayUtils.removeAll([2, 6, 3], 0, 1, 2) = []
  4789.      * </pre>
  4790.      *
  4791.      * @param array   the array to remove the element from, may not be {@code null}
  4792.      * @param indices the positions of the elements to be removed
  4793.      * @return A new array containing the existing elements except those
  4794.      *         at the specified positions.
  4795.      * @throws IndexOutOfBoundsException if any index is out of range
  4796.      * (index &lt; 0 || index &gt;= array.length), or if the array is {@code null}.
  4797.      * @since 3.0.1
  4798.      */
  4799.     public static byte[] removeAll(final byte[] array, final int... indices) {
  4800.         return (byte[]) removeAll((Object) array, indices);
  4801.     }

  4802.     /**
  4803.      * Removes the elements at the specified positions from the specified array.
  4804.      * All remaining elements are shifted to the left.
  4805.      * <p>
  4806.      * This method returns a new array with the same elements of the input
  4807.      * array except those at the specified positions. The component
  4808.      * type of the returned array is always the same as that of the input
  4809.      * array.
  4810.      * </p>
  4811.      * <p>
  4812.      * If the input array is {@code null}, an IndexOutOfBoundsException
  4813.      * will be thrown, because in that case no valid index can be specified.
  4814.      * </p>
  4815.      * <pre>
  4816.      * ArrayUtils.removeAll([1], 0)             = []
  4817.      * ArrayUtils.removeAll([2, 6], 0)          = [6]
  4818.      * ArrayUtils.removeAll([2, 6], 0, 1)       = []
  4819.      * ArrayUtils.removeAll([2, 6, 3], 1, 2)    = [2]
  4820.      * ArrayUtils.removeAll([2, 6, 3], 0, 2)    = [6]
  4821.      * ArrayUtils.removeAll([2, 6, 3], 0, 1, 2) = []
  4822.      * </pre>
  4823.      *
  4824.      * @param array   the array to remove the element from, may not be {@code null}
  4825.      * @param indices the positions of the elements to be removed
  4826.      * @return A new array containing the existing elements except those
  4827.      *         at the specified positions.
  4828.      * @throws IndexOutOfBoundsException if any index is out of range
  4829.      * (index &lt; 0 || index &gt;= array.length), or if the array is {@code null}.
  4830.      * @since 3.0.1
  4831.      */
  4832.     public static char[] removeAll(final char[] array, final int... indices) {
  4833.         return (char[]) removeAll((Object) array, indices);
  4834.     }

  4835.     /**
  4836.      * Removes the elements at the specified positions from the specified array.
  4837.      * All remaining elements are shifted to the left.
  4838.      * <p>
  4839.      * This method returns a new array with the same elements of the input
  4840.      * array except those at the specified positions. The component
  4841.      * type of the returned array is always the same as that of the input
  4842.      * array.
  4843.      * </p>
  4844.      * <p>
  4845.      * If the input array is {@code null}, an IndexOutOfBoundsException
  4846.      * will be thrown, because in that case no valid index can be specified.
  4847.      * </p>
  4848.      * <pre>
  4849.      * ArrayUtils.removeAll([1], 0)             = []
  4850.      * ArrayUtils.removeAll([2, 6], 0)          = [6]
  4851.      * ArrayUtils.removeAll([2, 6], 0, 1)       = []
  4852.      * ArrayUtils.removeAll([2, 6, 3], 1, 2)    = [2]
  4853.      * ArrayUtils.removeAll([2, 6, 3], 0, 2)    = [6]
  4854.      * ArrayUtils.removeAll([2, 6, 3], 0, 1, 2) = []
  4855.      * </pre>
  4856.      *
  4857.      * @param array   the array to remove the element from, may not be {@code null}
  4858.      * @param indices the positions of the elements to be removed
  4859.      * @return A new array containing the existing elements except those
  4860.      *         at the specified positions.
  4861.      * @throws IndexOutOfBoundsException if any index is out of range
  4862.      * (index &lt; 0 || index &gt;= array.length), or if the array is {@code null}.
  4863.      * @since 3.0.1
  4864.      */
  4865.     public static double[] removeAll(final double[] array, final int... indices) {
  4866.         return (double[]) removeAll((Object) array, indices);
  4867.     }

  4868.     /**
  4869.      * Removes the elements at the specified positions from the specified array.
  4870.      * All remaining elements are shifted to the left.
  4871.      * <p>
  4872.      * This method returns a new array with the same elements of the input
  4873.      * array except those at the specified positions. The component
  4874.      * type of the returned array is always the same as that of the input
  4875.      * array.
  4876.      * </p>
  4877.      * <p>
  4878.      * If the input array is {@code null}, an IndexOutOfBoundsException
  4879.      * will be thrown, because in that case no valid index can be specified.
  4880.      * </p>
  4881.      * <pre>
  4882.      * ArrayUtils.removeAll([1], 0)             = []
  4883.      * ArrayUtils.removeAll([2, 6], 0)          = [6]
  4884.      * ArrayUtils.removeAll([2, 6], 0, 1)       = []
  4885.      * ArrayUtils.removeAll([2, 6, 3], 1, 2)    = [2]
  4886.      * ArrayUtils.removeAll([2, 6, 3], 0, 2)    = [6]
  4887.      * ArrayUtils.removeAll([2, 6, 3], 0, 1, 2) = []
  4888.      * </pre>
  4889.      *
  4890.      * @param array   the array to remove the element from, may not be {@code null}
  4891.      * @param indices the positions of the elements to be removed
  4892.      * @return A new array containing the existing elements except those
  4893.      *         at the specified positions.
  4894.      * @throws IndexOutOfBoundsException if any index is out of range
  4895.      * (index &lt; 0 || index &gt;= array.length), or if the array is {@code null}.
  4896.      * @since 3.0.1
  4897.      */
  4898.     public static float[] removeAll(final float[] array, final int... indices) {
  4899.         return (float[]) removeAll((Object) array, indices);
  4900.     }

  4901.     /**
  4902.      * Removes the elements at the specified positions from the specified array.
  4903.      * All remaining elements are shifted to the left.
  4904.      * <p>
  4905.      * This method returns a new array with the same elements of the input
  4906.      * array except those at the specified positions. The component
  4907.      * type of the returned array is always the same as that of the input
  4908.      * array.
  4909.      * </p>
  4910.      * <p>
  4911.      * If the input array is {@code null}, an IndexOutOfBoundsException
  4912.      * will be thrown, because in that case no valid index can be specified.
  4913.      * </p>
  4914.      * <pre>
  4915.      * ArrayUtils.removeAll([1], 0)             = []
  4916.      * ArrayUtils.removeAll([2, 6], 0)          = [6]
  4917.      * ArrayUtils.removeAll([2, 6], 0, 1)       = []
  4918.      * ArrayUtils.removeAll([2, 6, 3], 1, 2)    = [2]
  4919.      * ArrayUtils.removeAll([2, 6, 3], 0, 2)    = [6]
  4920.      * ArrayUtils.removeAll([2, 6, 3], 0, 1, 2) = []
  4921.      * </pre>
  4922.      *
  4923.      * @param array   the array to remove the element from, may not be {@code null}
  4924.      * @param indices the positions of the elements to be removed
  4925.      * @return A new array containing the existing elements except those
  4926.      *         at the specified positions.
  4927.      * @throws IndexOutOfBoundsException if any index is out of range
  4928.      * (index &lt; 0 || index &gt;= array.length), or if the array is {@code null}.
  4929.      * @since 3.0.1
  4930.      */
  4931.     public static int[] removeAll(final int[] array, final int... indices) {
  4932.         return (int[]) removeAll((Object) array, indices);
  4933.     }

  4934.     /**
  4935.      * Removes the elements at the specified positions from the specified array.
  4936.      * All remaining elements are shifted to the left.
  4937.      * <p>
  4938.      * This method returns a new array with the same elements of the input
  4939.      * array except those at the specified positions. The component
  4940.      * type of the returned array is always the same as that of the input
  4941.      * array.
  4942.      * </p>
  4943.      * <p>
  4944.      * If the input array is {@code null}, an IndexOutOfBoundsException
  4945.      * will be thrown, because in that case no valid index can be specified.
  4946.      * </p>
  4947.      * <pre>
  4948.      * ArrayUtils.removeAll([1], 0)             = []
  4949.      * ArrayUtils.removeAll([2, 6], 0)          = [6]
  4950.      * ArrayUtils.removeAll([2, 6], 0, 1)       = []
  4951.      * ArrayUtils.removeAll([2, 6, 3], 1, 2)    = [2]
  4952.      * ArrayUtils.removeAll([2, 6, 3], 0, 2)    = [6]
  4953.      * ArrayUtils.removeAll([2, 6, 3], 0, 1, 2) = []
  4954.      * </pre>
  4955.      *
  4956.      * @param array   the array to remove the element from, may not be {@code null}
  4957.      * @param indices the positions of the elements to be removed
  4958.      * @return A new array containing the existing elements except those
  4959.      *         at the specified positions.
  4960.      * @throws IndexOutOfBoundsException if any index is out of range
  4961.      * (index &lt; 0 || index &gt;= array.length), or if the array is {@code null}.
  4962.      * @since 3.0.1
  4963.      */
  4964.     public static long[] removeAll(final long[] array, final int... indices) {
  4965.         return (long[]) removeAll((Object) array, indices);
  4966.     }

  4967.     /**
  4968.      * Removes multiple array elements specified by indices.
  4969.      *
  4970.      * @param array source
  4971.      * @param indices to remove
  4972.      * @return new array of same type minus elements specified by the set bits in {@code indices}
  4973.      * @since 3.2
  4974.      */
  4975.     // package protected for access by unit tests
  4976.     static Object removeAll(final Object array, final BitSet indices) {
  4977.         if (array == null) {
  4978.             return null;
  4979.         }
  4980.         final int srcLength = getLength(array);
  4981.         // No need to check maxIndex here, because method only currently called from removeElements()
  4982.         // which guarantee to generate only valid bit entries.
  4983. //        final int maxIndex = indices.length();
  4984. //        if (maxIndex > srcLength) {
  4985. //            throw new IndexOutOfBoundsException("Index: " + (maxIndex-1) + ", Length: " + srcLength);
  4986. //        }
  4987.         final int removals = indices.cardinality(); // true bits are items to remove
  4988.         final Object result = Array.newInstance(array.getClass().getComponentType(), srcLength - removals);
  4989.         int srcIndex = 0;
  4990.         int destIndex = 0;
  4991.         int count;
  4992.         int set;
  4993.         while ((set = indices.nextSetBit(srcIndex)) != -1) {
  4994.             count = set - srcIndex;
  4995.             if (count > 0) {
  4996.                 System.arraycopy(array, srcIndex, result, destIndex, count);
  4997.                 destIndex += count;
  4998.             }
  4999.             srcIndex = indices.nextClearBit(set);
  5000.         }
  5001.         count = srcLength - srcIndex;
  5002.         if (count > 0) {
  5003.             System.arraycopy(array, srcIndex, result, destIndex, count);
  5004.         }
  5005.         return result;
  5006.     }

  5007.     /**
  5008.      * Removes multiple array elements specified by index.
  5009.      *
  5010.      * @param array source
  5011.      * @param indices to remove
  5012.      * @return new array of same type minus elements specified by unique values of {@code indices}
  5013.      */
  5014.     // package protected for access by unit tests
  5015.     static Object removeAll(final Object array, final int... indices) {
  5016.         if (array == null) {
  5017.             return null;
  5018.         }
  5019.         final int length = getLength(array);
  5020.         int diff = 0; // number of distinct indexes, i.e. number of entries that will be removed
  5021.         final int[] clonedIndices = ArraySorter.sort(clone(indices));
  5022.         // identify length of result array
  5023.         if (isNotEmpty(clonedIndices)) {
  5024.             int i = clonedIndices.length;
  5025.             int prevIndex = length;
  5026.             while (--i >= 0) {
  5027.                 final int index = clonedIndices[i];
  5028.                 if (index < 0 || index >= length) {
  5029.                     throw new IndexOutOfBoundsException("Index: " + index + ", Length: " + length);
  5030.                 }
  5031.                 if (index >= prevIndex) {
  5032.                     continue;
  5033.                 }
  5034.                 diff++;
  5035.                 prevIndex = index;
  5036.             }
  5037.         }
  5038.         // create result array
  5039.         final Object result = Array.newInstance(array.getClass().getComponentType(), length - diff);
  5040.         if (diff < length && clonedIndices != null) {
  5041.             int end = length; // index just after last copy
  5042.             int dest = length - diff; // number of entries so far not copied
  5043.             for (int i = clonedIndices.length - 1; i >= 0; i--) {
  5044.                 final int index = clonedIndices[i];
  5045.                 if (end - index > 1) { // same as (cp > 0)
  5046.                     final int cp = end - index - 1;
  5047.                     dest -= cp;
  5048.                     System.arraycopy(array, index + 1, result, dest, cp);
  5049.                     // After this copy, we still have room for dest items.
  5050.                 }
  5051.                 end = index;
  5052.             }
  5053.             if (end > 0) {
  5054.                 System.arraycopy(array, 0, result, 0, end);
  5055.             }
  5056.         }
  5057.         return result;
  5058.     }

  5059.     /**
  5060.      * Removes the elements at the specified positions from the specified array.
  5061.      * All remaining elements are shifted to the left.
  5062.      * <p>
  5063.      * This method returns a new array with the same elements of the input
  5064.      * array except those at the specified positions. The component
  5065.      * type of the returned array is always the same as that of the input
  5066.      * array.
  5067.      * </p>
  5068.      * <p>
  5069.      * If the input array is {@code null}, an IndexOutOfBoundsException
  5070.      * will be thrown, because in that case no valid index can be specified.
  5071.      * </p>
  5072.      * <pre>
  5073.      * ArrayUtils.removeAll([1], 0)             = []
  5074.      * ArrayUtils.removeAll([2, 6], 0)          = [6]
  5075.      * ArrayUtils.removeAll([2, 6], 0, 1)       = []
  5076.      * ArrayUtils.removeAll([2, 6, 3], 1, 2)    = [2]
  5077.      * ArrayUtils.removeAll([2, 6, 3], 0, 2)    = [6]
  5078.      * ArrayUtils.removeAll([2, 6, 3], 0, 1, 2) = []
  5079.      * </pre>
  5080.      *
  5081.      * @param array   the array to remove the element from, may not be {@code null}
  5082.      * @param indices the positions of the elements to be removed
  5083.      * @return A new array containing the existing elements except those
  5084.      *         at the specified positions.
  5085.      * @throws IndexOutOfBoundsException if any index is out of range
  5086.      * (index &lt; 0 || index &gt;= array.length), or if the array is {@code null}.
  5087.      * @since 3.0.1
  5088.      */
  5089.     public static short[] removeAll(final short[] array, final int... indices) {
  5090.         return (short[]) removeAll((Object) array, indices);
  5091.     }

  5092.     /**
  5093.      * Removes the elements at the specified positions from the specified array.
  5094.      * All remaining elements are shifted to the left.
  5095.      * <p>
  5096.      * This method returns a new array with the same elements of the input
  5097.      * array except those at the specified positions. The component
  5098.      * type of the returned array is always the same as that of the input
  5099.      * array.
  5100.      * </p>
  5101.      * <p>
  5102.      * If the input array is {@code null}, an IndexOutOfBoundsException
  5103.      * will be thrown, because in that case no valid index can be specified.
  5104.      * </p>
  5105.      * <pre>
  5106.      * ArrayUtils.removeAll(["a", "b", "c"], 0, 2) = ["b"]
  5107.      * ArrayUtils.removeAll(["a", "b", "c"], 1, 2) = ["a"]
  5108.      * </pre>
  5109.      *
  5110.      * @param <T> the component type of the array
  5111.      * @param array   the array to remove the element from, may not be {@code null}
  5112.      * @param indices the positions of the elements to be removed
  5113.      * @return A new array containing the existing elements except those
  5114.      *         at the specified positions.
  5115.      * @throws IndexOutOfBoundsException if any index is out of range
  5116.      * (index &lt; 0 || index &gt;= array.length), or if the array is {@code null}.
  5117.      * @since 3.0.1
  5118.      */
  5119.     @SuppressWarnings("unchecked") // removeAll() always creates an array of the same type as its input
  5120.     public static <T> T[] removeAll(final T[] array, final int... indices) {
  5121.         return (T[]) removeAll((Object) array, indices);
  5122.     }

  5123.     /**
  5124.      * Removes the occurrences of the specified element from the specified boolean array.
  5125.      * <p>
  5126.      * All subsequent elements are shifted to the left (subtracts one from their indices).
  5127.      * If the array doesn't contain such an element, no elements are removed from the array.
  5128.      * {@code null} will be returned if the input array is {@code null}.
  5129.      * </p>
  5130.      *
  5131.      * @param element the element to remove
  5132.      * @param array the input array
  5133.      *
  5134.      * @return A new array containing the existing elements except the occurrences of the specified element.
  5135.      * @since 3.5
  5136.      * @deprecated Use {@link #removeAllOccurrences(boolean[], boolean)}
  5137.      */
  5138.     @Deprecated
  5139.     public static boolean[] removeAllOccurences(final boolean[] array, final boolean element) {
  5140.         return (boolean[]) removeAll((Object) array, indexesOf(array, element));
  5141.     }

  5142.     /**
  5143.      * Removes the occurrences of the specified element from the specified byte array.
  5144.      * <p>
  5145.      * All subsequent elements are shifted to the left (subtracts one from their indices).
  5146.      * If the array doesn't contain such an element, no elements are removed from the array.
  5147.      * {@code null} will be returned if the input array is {@code null}.
  5148.      * </p>
  5149.      *
  5150.      * @param element the element to remove
  5151.      * @param array the input array
  5152.      *
  5153.      * @return A new array containing the existing elements except the occurrences of the specified element.
  5154.      * @since 3.5
  5155.      * @deprecated Use {@link #removeAllOccurrences(byte[], byte)}
  5156.      */
  5157.     @Deprecated
  5158.     public static byte[] removeAllOccurences(final byte[] array, final byte element) {
  5159.         return (byte[]) removeAll((Object) array, indexesOf(array, element));
  5160.     }

  5161.     /**
  5162.      * Removes the occurrences of the specified element from the specified char array.
  5163.      * <p>
  5164.      * All subsequent elements are shifted to the left (subtracts one from their indices).
  5165.      * If the array doesn't contain such an element, no elements are removed from the array.
  5166.      * {@code null} will be returned if the input array is {@code null}.
  5167.      * </p>
  5168.      *
  5169.      * @param element the element to remove
  5170.      * @param array the input array
  5171.      *
  5172.      * @return A new array containing the existing elements except the occurrences of the specified element.
  5173.      * @since 3.5
  5174.      * @deprecated Use {@link #removeAllOccurrences(char[], char)}
  5175.      */
  5176.     @Deprecated
  5177.     public static char[] removeAllOccurences(final char[] array, final char element) {
  5178.         return (char[]) removeAll((Object) array, indexesOf(array, element));
  5179.     }

  5180.     /**
  5181.      * Removes the occurrences of the specified element from the specified double array.
  5182.      * <p>
  5183.      * All subsequent elements are shifted to the left (subtracts one from their indices).
  5184.      * If the array doesn't contain such an element, no elements are removed from the array.
  5185.      * {@code null} will be returned if the input array is {@code null}.
  5186.      * </p>
  5187.      *
  5188.      * @param element the element to remove
  5189.      * @param array the input array
  5190.      *
  5191.      * @return A new array containing the existing elements except the occurrences of the specified element.
  5192.      * @since 3.5
  5193.      * @deprecated Use {@link #removeAllOccurrences(double[], double)}
  5194.      */
  5195.     @Deprecated
  5196.     public static double[] removeAllOccurences(final double[] array, final double element) {
  5197.         return (double[]) removeAll((Object) array, indexesOf(array, element));
  5198.     }

  5199.     /**
  5200.      * Removes the occurrences of the specified element from the specified float array.
  5201.      * <p>
  5202.      * All subsequent elements are shifted to the left (subtracts one from their indices).
  5203.      * If the array doesn't contain such an element, no elements are removed from the array.
  5204.      * {@code null} will be returned if the input array is {@code null}.
  5205.      * </p>
  5206.      *
  5207.      * @param element the element to remove
  5208.      * @param array the input array
  5209.      *
  5210.      * @return A new array containing the existing elements except the occurrences of the specified element.
  5211.      * @since 3.5
  5212.      * @deprecated Use {@link #removeAllOccurrences(float[], float)}
  5213.      */
  5214.     @Deprecated
  5215.     public static float[] removeAllOccurences(final float[] array, final float element) {
  5216.         return (float[]) removeAll((Object) array, indexesOf(array, element));
  5217.     }

  5218.     /**
  5219.      * Removes the occurrences of the specified element from the specified int array.
  5220.      * <p>
  5221.      * All subsequent elements are shifted to the left (subtracts one from their indices).
  5222.      * If the array doesn't contain such an element, no elements are removed from the array.
  5223.      * {@code null} will be returned if the input array is {@code null}.
  5224.      * </p>
  5225.      *
  5226.      * @param element the element to remove
  5227.      * @param array the input array
  5228.      *
  5229.      * @return A new array containing the existing elements except the occurrences of the specified element.
  5230.      * @since 3.5
  5231.      * @deprecated Use {@link #removeAllOccurrences(int[], int)}
  5232.      */
  5233.     @Deprecated
  5234.     public static int[] removeAllOccurences(final int[] array, final int element) {
  5235.         return (int[]) removeAll((Object) array, indexesOf(array, element));
  5236.     }

  5237.     /**
  5238.      * Removes the occurrences of the specified element from the specified long array.
  5239.      * <p>
  5240.      * All subsequent elements are shifted to the left (subtracts one from their indices).
  5241.      * If the array doesn't contain such an element, no elements are removed from the array.
  5242.      * {@code null} will be returned if the input array is {@code null}.
  5243.      * </p>
  5244.      *
  5245.      * @param element the element to remove
  5246.      * @param array the input array
  5247.      *
  5248.      * @return A new array containing the existing elements except the occurrences of the specified element.
  5249.      * @since 3.5
  5250.      * @deprecated Use {@link #removeAllOccurrences(long[], long)}
  5251.      */
  5252.     @Deprecated
  5253.     public static long[] removeAllOccurences(final long[] array, final long element) {
  5254.         return (long[]) removeAll((Object) array, indexesOf(array, element));
  5255.     }

  5256.     /**
  5257.      * Removes the occurrences of the specified element from the specified short array.
  5258.      * <p>
  5259.      * All subsequent elements are shifted to the left (subtracts one from their indices).
  5260.      * If the array doesn't contain such an element, no elements are removed from the array.
  5261.      * {@code null} will be returned if the input array is {@code null}.
  5262.      * </p>
  5263.      *
  5264.      * @param element the element to remove
  5265.      * @param array the input array
  5266.      *
  5267.      * @return A new array containing the existing elements except the occurrences of the specified element.
  5268.      * @since 3.5
  5269.      * @deprecated Use {@link #removeAllOccurrences(short[], short)}
  5270.      */
  5271.     @Deprecated
  5272.     public static short[] removeAllOccurences(final short[] array, final short element) {
  5273.         return (short[]) removeAll((Object) array, indexesOf(array, element));
  5274.     }

  5275.     /**
  5276.      * Removes the occurrences of the specified element from the specified array.
  5277.      * <p>
  5278.      * All subsequent elements are shifted to the left (subtracts one from their indices).
  5279.      * If the array doesn't contain such an element, no elements are removed from the array.
  5280.      * {@code null} will be returned if the input array is {@code null}.
  5281.      * </p>
  5282.      *
  5283.      * @param <T> the type of object in the array
  5284.      * @param element the element to remove
  5285.      * @param array the input array
  5286.      *
  5287.      * @return A new array containing the existing elements except the occurrences of the specified element.
  5288.      * @since 3.5
  5289.      * @deprecated Use {@link #removeAllOccurrences(Object[], Object)}
  5290.      */
  5291.     @Deprecated
  5292.     public static <T> T[] removeAllOccurences(final T[] array, final T element) {
  5293.         return (T[]) removeAll((Object) array, indexesOf(array, element));
  5294.     }

  5295.     /**
  5296.      * Removes the occurrences of the specified element from the specified boolean array.
  5297.      * <p>
  5298.      * All subsequent elements are shifted to the left (subtracts one from their indices).
  5299.      * If the array doesn't contain such an element, no elements are removed from the array.
  5300.      * {@code null} will be returned if the input array is {@code null}.
  5301.      * </p>
  5302.      *
  5303.      * @param element the element to remove
  5304.      * @param array the input array
  5305.      *
  5306.      * @return A new array containing the existing elements except the occurrences of the specified element.
  5307.      * @since 3.10
  5308.      */
  5309.     public static boolean[] removeAllOccurrences(final boolean[] array, final boolean element) {
  5310.         return (boolean[]) removeAll((Object) array, indexesOf(array, element));
  5311.     }

  5312.     /**
  5313.      * Removes the occurrences of the specified element from the specified byte array.
  5314.      * <p>
  5315.      * All subsequent elements are shifted to the left (subtracts one from their indices).
  5316.      * If the array doesn't contain such an element, no elements are removed from the array.
  5317.      * {@code null} will be returned if the input array is {@code null}.
  5318.      * </p>
  5319.      *
  5320.      * @param element the element to remove
  5321.      * @param array the input array
  5322.      *
  5323.      * @return A new array containing the existing elements except the occurrences of the specified element.
  5324.      * @since 3.10
  5325.      */
  5326.     public static byte[] removeAllOccurrences(final byte[] array, final byte element) {
  5327.         return (byte[]) removeAll((Object) array, indexesOf(array, element));
  5328.     }

  5329.     /**
  5330.      * Removes the occurrences of the specified element from the specified char array.
  5331.      * <p>
  5332.      * All subsequent elements are shifted to the left (subtracts one from their indices).
  5333.      * If the array doesn't contain such an element, no elements are removed from the array.
  5334.      * {@code null} will be returned if the input array is {@code null}.
  5335.      * </p>
  5336.      *
  5337.      * @param element the element to remove
  5338.      * @param array the input array
  5339.      *
  5340.      * @return A new array containing the existing elements except the occurrences of the specified element.
  5341.      * @since 3.10
  5342.      */
  5343.     public static char[] removeAllOccurrences(final char[] array, final char element) {
  5344.         return (char[]) removeAll((Object) array, indexesOf(array, element));
  5345.     }

  5346.     /**
  5347.      * Removes the occurrences of the specified element from the specified double array.
  5348.      * <p>
  5349.      * All subsequent elements are shifted to the left (subtracts one from their indices).
  5350.      * If the array doesn't contain such an element, no elements are removed from the array.
  5351.      * {@code null} will be returned if the input array is {@code null}.
  5352.      * </p>
  5353.      *
  5354.      * @param element the element to remove
  5355.      * @param array the input array
  5356.      *
  5357.      * @return A new array containing the existing elements except the occurrences of the specified element.
  5358.      * @since 3.10
  5359.      */
  5360.     public static double[] removeAllOccurrences(final double[] array, final double element) {
  5361.         return (double[]) removeAll((Object) array, indexesOf(array, element));
  5362.     }

  5363.     /**
  5364.      * Removes the occurrences of the specified element from the specified float array.
  5365.      * <p>
  5366.      * All subsequent elements are shifted to the left (subtracts one from their indices).
  5367.      * If the array doesn't contain such an element, no elements are removed from the array.
  5368.      * {@code null} will be returned if the input array is {@code null}.
  5369.      * </p>
  5370.      *
  5371.      * @param element the element to remove
  5372.      * @param array the input array
  5373.      *
  5374.      * @return A new array containing the existing elements except the occurrences of the specified element.
  5375.      * @since 3.10
  5376.      */
  5377.     public static float[] removeAllOccurrences(final float[] array, final float element) {
  5378.         return (float[]) removeAll((Object) array, indexesOf(array, element));
  5379.     }

  5380.     /**
  5381.      * Removes the occurrences of the specified element from the specified int array.
  5382.      * <p>
  5383.      * All subsequent elements are shifted to the left (subtracts one from their indices).
  5384.      * If the array doesn't contain such an element, no elements are removed from the array.
  5385.      * {@code null} will be returned if the input array is {@code null}.
  5386.      * </p>
  5387.      *
  5388.      * @param element the element to remove
  5389.      * @param array the input array
  5390.      *
  5391.      * @return A new array containing the existing elements except the occurrences of the specified element.
  5392.      * @since 3.10
  5393.      */
  5394.     public static int[] removeAllOccurrences(final int[] array, final int element) {
  5395.         return (int[]) removeAll((Object) array, indexesOf(array, element));
  5396.     }

  5397.     /**
  5398.      * Removes the occurrences of the specified element from the specified long array.
  5399.      * <p>
  5400.      * All subsequent elements are shifted to the left (subtracts one from their indices).
  5401.      * If the array doesn't contain such an element, no elements are removed from the array.
  5402.      * {@code null} will be returned if the input array is {@code null}.
  5403.      * </p>
  5404.      *
  5405.      * @param element the element to remove
  5406.      * @param array the input array
  5407.      *
  5408.      * @return A new array containing the existing elements except the occurrences of the specified element.
  5409.      * @since 3.10
  5410.      */
  5411.     public static long[] removeAllOccurrences(final long[] array, final long element) {
  5412.         return (long[]) removeAll((Object) array, indexesOf(array, element));
  5413.     }

  5414.     /**
  5415.      * Removes the occurrences of the specified element from the specified short array.
  5416.      * <p>
  5417.      * All subsequent elements are shifted to the left (subtracts one from their indices).
  5418.      * If the array doesn't contain such an element, no elements are removed from the array.
  5419.      * {@code null} will be returned if the input array is {@code null}.
  5420.      * </p>
  5421.      *
  5422.      * @param element the element to remove
  5423.      * @param array the input array
  5424.      *
  5425.      * @return A new array containing the existing elements except the occurrences of the specified element.
  5426.      * @since 3.10
  5427.      */
  5428.     public static short[] removeAllOccurrences(final short[] array, final short element) {
  5429.         return (short[]) removeAll((Object) array, indexesOf(array, element));
  5430.     }

  5431.     /**
  5432.      * Removes the occurrences of the specified element from the specified array.
  5433.      * <p>
  5434.      * All subsequent elements are shifted to the left (subtracts one from their indices).
  5435.      * If the array doesn't contain such an element, no elements are removed from the array.
  5436.      * {@code null} will be returned if the input array is {@code null}.
  5437.      * </p>
  5438.      *
  5439.      * @param <T> the type of object in the array
  5440.      * @param element the element to remove
  5441.      * @param array the input array
  5442.      *
  5443.      * @return A new array containing the existing elements except the occurrences of the specified element.
  5444.      * @since 3.10
  5445.      */
  5446.     public static <T> T[] removeAllOccurrences(final T[] array, final T element) {
  5447.         return (T[]) removeAll((Object) array, indexesOf(array, element));
  5448.     }

  5449.     /**
  5450.      * Removes the first occurrence of the specified element from the
  5451.      * specified array. All subsequent elements are shifted to the left
  5452.      * (subtracts one from their indices). If the array doesn't contain
  5453.      * such an element, no elements are removed from the array.
  5454.      * <p>
  5455.      * This method returns a new array with the same elements of the input
  5456.      * array except the first occurrence of the specified element. The component
  5457.      * type of the returned array is always the same as that of the input
  5458.      * array.
  5459.      * </p>
  5460.      * <pre>
  5461.      * ArrayUtils.removeElement(null, true)                = null
  5462.      * ArrayUtils.removeElement([], true)                  = []
  5463.      * ArrayUtils.removeElement([true], false)             = [true]
  5464.      * ArrayUtils.removeElement([true, false], false)      = [true]
  5465.      * ArrayUtils.removeElement([true, false, true], true) = [false, true]
  5466.      * </pre>
  5467.      *
  5468.      * @param array  the array to remove the element from, may be {@code null}
  5469.      * @param element  the element to be removed
  5470.      * @return A new array containing the existing elements except the first
  5471.      *         occurrence of the specified element.
  5472.      * @since 2.1
  5473.      */
  5474.     public static boolean[] removeElement(final boolean[] array, final boolean element) {
  5475.         final int index = indexOf(array, element);
  5476.         return index == INDEX_NOT_FOUND ? clone(array) : remove(array, index);
  5477.     }

  5478.     /**
  5479.      * Removes the first occurrence of the specified element from the
  5480.      * specified array. All subsequent elements are shifted to the left
  5481.      * (subtracts one from their indices). If the array doesn't contain
  5482.      * such an element, no elements are removed from the array.
  5483.      * <p>
  5484.      * This method returns a new array with the same elements of the input
  5485.      * array except the first occurrence of the specified element. The component
  5486.      * type of the returned array is always the same as that of the input
  5487.      * array.
  5488.      * </p>
  5489.      * <pre>
  5490.      * ArrayUtils.removeElement(null, 1)        = null
  5491.      * ArrayUtils.removeElement([], 1)          = []
  5492.      * ArrayUtils.removeElement([1], 0)         = [1]
  5493.      * ArrayUtils.removeElement([1, 0], 0)      = [1]
  5494.      * ArrayUtils.removeElement([1, 0, 1], 1)   = [0, 1]
  5495.      * </pre>
  5496.      *
  5497.      * @param array  the array to remove the element from, may be {@code null}
  5498.      * @param element  the element to be removed
  5499.      * @return A new array containing the existing elements except the first
  5500.      *         occurrence of the specified element.
  5501.      * @since 2.1
  5502.      */
  5503.     public static byte[] removeElement(final byte[] array, final byte element) {
  5504.         final int index = indexOf(array, element);
  5505.         return index == INDEX_NOT_FOUND ? clone(array) : remove(array, index);
  5506.     }

  5507.     /**
  5508.      * Removes the first occurrence of the specified element from the
  5509.      * specified array. All subsequent elements are shifted to the left
  5510.      * (subtracts one from their indices). If the array doesn't contain
  5511.      * such an element, no elements are removed from the array.
  5512.      * <p>
  5513.      * This method returns a new array with the same elements of the input
  5514.      * array except the first occurrence of the specified element. The component
  5515.      * type of the returned array is always the same as that of the input
  5516.      * array.
  5517.      * </p>
  5518.      * <pre>
  5519.      * ArrayUtils.removeElement(null, 'a')            = null
  5520.      * ArrayUtils.removeElement([], 'a')              = []
  5521.      * ArrayUtils.removeElement(['a'], 'b')           = ['a']
  5522.      * ArrayUtils.removeElement(['a', 'b'], 'a')      = ['b']
  5523.      * ArrayUtils.removeElement(['a', 'b', 'a'], 'a') = ['b', 'a']
  5524.      * </pre>
  5525.      *
  5526.      * @param array  the array to remove the element from, may be {@code null}
  5527.      * @param element  the element to be removed
  5528.      * @return A new array containing the existing elements except the first
  5529.      *         occurrence of the specified element.
  5530.      * @since 2.1
  5531.      */
  5532.     public static char[] removeElement(final char[] array, final char element) {
  5533.         final int index = indexOf(array, element);
  5534.         return index == INDEX_NOT_FOUND ? clone(array) : remove(array, index);
  5535.     }

  5536.     /**
  5537.      * Removes the first occurrence of the specified element from the
  5538.      * specified array. All subsequent elements are shifted to the left
  5539.      * (subtracts one from their indices). If the array doesn't contain
  5540.      * such an element, no elements are removed from the array.
  5541.      * <p>
  5542.      * This method returns a new array with the same elements of the input
  5543.      * array except the first occurrence of the specified element. The component
  5544.      * type of the returned array is always the same as that of the input
  5545.      * array.
  5546.      * </p>
  5547.      * <pre>
  5548.      * ArrayUtils.removeElement(null, 1.1)            = null
  5549.      * ArrayUtils.removeElement([], 1.1)              = []
  5550.      * ArrayUtils.removeElement([1.1], 1.2)           = [1.1]
  5551.      * ArrayUtils.removeElement([1.1, 2.3], 1.1)      = [2.3]
  5552.      * ArrayUtils.removeElement([1.1, 2.3, 1.1], 1.1) = [2.3, 1.1]
  5553.      * </pre>
  5554.      *
  5555.      * @param array  the array to remove the element from, may be {@code null}
  5556.      * @param element  the element to be removed
  5557.      * @return A new array containing the existing elements except the first
  5558.      *         occurrence of the specified element.
  5559.      * @since 2.1
  5560.      */
  5561.     public static double[] removeElement(final double[] array, final double element) {
  5562.         final int index = indexOf(array, element);
  5563.         return index == INDEX_NOT_FOUND ? clone(array) : remove(array, index);
  5564.     }

  5565.     /**
  5566.      * Removes the first occurrence of the specified element from the
  5567.      * specified array. All subsequent elements are shifted to the left
  5568.      * (subtracts one from their indices). If the array doesn't contain
  5569.      * such an element, no elements are removed from the array.
  5570.      * <p>
  5571.      * This method returns a new array with the same elements of the input
  5572.      * array except the first occurrence of the specified element. The component
  5573.      * type of the returned array is always the same as that of the input
  5574.      * array.
  5575.      * </p>
  5576.      * <pre>
  5577.      * ArrayUtils.removeElement(null, 1.1)            = null
  5578.      * ArrayUtils.removeElement([], 1.1)              = []
  5579.      * ArrayUtils.removeElement([1.1], 1.2)           = [1.1]
  5580.      * ArrayUtils.removeElement([1.1, 2.3], 1.1)      = [2.3]
  5581.      * ArrayUtils.removeElement([1.1, 2.3, 1.1], 1.1) = [2.3, 1.1]
  5582.      * </pre>
  5583.      *
  5584.      * @param array  the array to remove the element from, may be {@code null}
  5585.      * @param element  the element to be removed
  5586.      * @return A new array containing the existing elements except the first
  5587.      *         occurrence of the specified element.
  5588.      * @since 2.1
  5589.      */
  5590.     public static float[] removeElement(final float[] array, final float element) {
  5591.         final int index = indexOf(array, element);
  5592.         return index == INDEX_NOT_FOUND ? clone(array) : remove(array, index);
  5593.     }

  5594.     /**
  5595.      * Removes the first occurrence of the specified element from the
  5596.      * specified array. All subsequent elements are shifted to the left
  5597.      * (subtracts one from their indices). If the array doesn't contain
  5598.      * such an element, no elements are removed from the array.
  5599.      * <p>
  5600.      * This method returns a new array with the same elements of the input
  5601.      * array except the first occurrence of the specified element. The component
  5602.      * type of the returned array is always the same as that of the input
  5603.      * array.
  5604.      * </p>
  5605.      * <pre>
  5606.      * ArrayUtils.removeElement(null, 1)      = null
  5607.      * ArrayUtils.removeElement([], 1)        = []
  5608.      * ArrayUtils.removeElement([1], 2)       = [1]
  5609.      * ArrayUtils.removeElement([1, 3], 1)    = [3]
  5610.      * ArrayUtils.removeElement([1, 3, 1], 1) = [3, 1]
  5611.      * </pre>
  5612.      *
  5613.      * @param array  the array to remove the element from, may be {@code null}
  5614.      * @param element  the element to be removed
  5615.      * @return A new array containing the existing elements except the first
  5616.      *         occurrence of the specified element.
  5617.      * @since 2.1
  5618.      */
  5619.     public static int[] removeElement(final int[] array, final int element) {
  5620.         final int index = indexOf(array, element);
  5621.         return index == INDEX_NOT_FOUND ? clone(array) : remove(array, index);
  5622.     }

  5623.     /**
  5624.      * Removes the first occurrence of the specified element from the
  5625.      * specified array. All subsequent elements are shifted to the left
  5626.      * (subtracts one from their indices). If the array doesn't contain
  5627.      * such an element, no elements are removed from the array.
  5628.      * <p>
  5629.      * This method returns a new array with the same elements of the input
  5630.      * array except the first occurrence of the specified element. The component
  5631.      * type of the returned array is always the same as that of the input
  5632.      * array.
  5633.      * </p>
  5634.      * <pre>
  5635.      * ArrayUtils.removeElement(null, 1)      = null
  5636.      * ArrayUtils.removeElement([], 1)        = []
  5637.      * ArrayUtils.removeElement([1], 2)       = [1]
  5638.      * ArrayUtils.removeElement([1, 3], 1)    = [3]
  5639.      * ArrayUtils.removeElement([1, 3, 1], 1) = [3, 1]
  5640.      * </pre>
  5641.      *
  5642.      * @param array  the array to remove the element from, may be {@code null}
  5643.      * @param element  the element to be removed
  5644.      * @return A new array containing the existing elements except the first
  5645.      *         occurrence of the specified element.
  5646.      * @since 2.1
  5647.      */
  5648.     public static long[] removeElement(final long[] array, final long element) {
  5649.         final int index = indexOf(array, element);
  5650.         return index == INDEX_NOT_FOUND ? clone(array) : remove(array, index);
  5651.     }

  5652.     /**
  5653.      * Removes the first occurrence of the specified element from the
  5654.      * specified array. All subsequent elements are shifted to the left
  5655.      * (subtracts one from their indices). If the array doesn't contain
  5656.      * such an element, no elements are removed from the array.
  5657.      * <p>
  5658.      * This method returns a new array with the same elements of the input
  5659.      * array except the first occurrence of the specified element. The component
  5660.      * type of the returned array is always the same as that of the input
  5661.      * array.
  5662.      * </p>
  5663.      * <pre>
  5664.      * ArrayUtils.removeElement(null, 1)      = null
  5665.      * ArrayUtils.removeElement([], 1)        = []
  5666.      * ArrayUtils.removeElement([1], 2)       = [1]
  5667.      * ArrayUtils.removeElement([1, 3], 1)    = [3]
  5668.      * ArrayUtils.removeElement([1, 3, 1], 1) = [3, 1]
  5669.      * </pre>
  5670.      *
  5671.      * @param array  the array to remove the element from, may be {@code null}
  5672.      * @param element  the element to be removed
  5673.      * @return A new array containing the existing elements except the first
  5674.      *         occurrence of the specified element.
  5675.      * @since 2.1
  5676.      */
  5677.     public static short[] removeElement(final short[] array, final short element) {
  5678.         final int index = indexOf(array, element);
  5679.         return index == INDEX_NOT_FOUND ? clone(array) : remove(array, index);
  5680.     }

  5681.     /**
  5682.      * Removes the first occurrence of the specified element from the
  5683.      * specified array. All subsequent elements are shifted to the left
  5684.      * (subtracts one from their indices). If the array doesn't contain
  5685.      * such an element, no elements are removed from the array.
  5686.      * <p>
  5687.      * This method returns a new array with the same elements of the input
  5688.      * array except the first occurrence of the specified element. The component
  5689.      * type of the returned array is always the same as that of the input
  5690.      * array.
  5691.      * </p>
  5692.      * <pre>
  5693.      * ArrayUtils.removeElement(null, "a")            = null
  5694.      * ArrayUtils.removeElement([], "a")              = []
  5695.      * ArrayUtils.removeElement(["a"], "b")           = ["a"]
  5696.      * ArrayUtils.removeElement(["a", "b"], "a")      = ["b"]
  5697.      * ArrayUtils.removeElement(["a", "b", "a"], "a") = ["b", "a"]
  5698.      * </pre>
  5699.      *
  5700.      * @param <T> the component type of the array
  5701.      * @param array  the array to remove the element from, may be {@code null}
  5702.      * @param element  the element to be removed
  5703.      * @return A new array containing the existing elements except the first
  5704.      *         occurrence of the specified element.
  5705.      * @since 2.1
  5706.      */
  5707.     public static <T> T[] removeElement(final T[] array, final Object element) {
  5708.         final int index = indexOf(array, element);
  5709.         return index == INDEX_NOT_FOUND ? clone(array) : remove(array, index);
  5710.     }

  5711.     /**
  5712.      * Removes occurrences of specified elements, in specified quantities,
  5713.      * from the specified array. All subsequent elements are shifted left.
  5714.      * For any element-to-be-removed specified in greater quantities than
  5715.      * contained in the original array, no change occurs beyond the
  5716.      * removal of the existing matching items.
  5717.      * <p>
  5718.      * This method returns a new array with the same elements of the input
  5719.      * array except for the earliest-encountered occurrences of the specified
  5720.      * elements. The component type of the returned array is always the same
  5721.      * as that of the input array.
  5722.      * </p>
  5723.      * <pre>
  5724.      * ArrayUtils.removeElements(null, true, false)               = null
  5725.      * ArrayUtils.removeElements([], true, false)                 = []
  5726.      * ArrayUtils.removeElements([true], false, false)            = [true]
  5727.      * ArrayUtils.removeElements([true, false], true, true)       = [false]
  5728.      * ArrayUtils.removeElements([true, false, true], true)       = [false, true]
  5729.      * ArrayUtils.removeElements([true, false, true], true, true) = [false]
  5730.      * </pre>
  5731.      *
  5732.      * @param array  the array to remove the element from, may be {@code null}
  5733.      * @param values the elements to be removed
  5734.      * @return A new array containing the existing elements except the
  5735.      *         earliest-encountered occurrences of the specified elements.
  5736.      * @since 3.0.1
  5737.      */
  5738.     public static boolean[] removeElements(final boolean[] array, final boolean... values) {
  5739.         if (isEmpty(array) || isEmpty(values)) {
  5740.             return clone(array);
  5741.         }
  5742.         final HashMap<Boolean, MutableInt> occurrences = new HashMap<>(2); // only two possible values here
  5743.         for (final boolean v : values) {
  5744.             final Boolean boxed = Boolean.valueOf(v);
  5745.             final MutableInt count = occurrences.get(boxed);
  5746.             if (count == null) {
  5747.                 occurrences.put(boxed, new MutableInt(1));
  5748.             } else {
  5749.                 count.increment();
  5750.             }
  5751.         }
  5752.         final BitSet toRemove = new BitSet();
  5753.         for (int i = 0; i < array.length; i++) {
  5754.             final boolean key = array[i];
  5755.             final MutableInt count = occurrences.get(key);
  5756.             if (count != null) {
  5757.                 if (count.decrementAndGet() == 0) {
  5758.                     occurrences.remove(key);
  5759.                 }
  5760.                 toRemove.set(i);
  5761.             }
  5762.         }
  5763.         return (boolean[]) removeAll(array, toRemove);
  5764.     }

  5765.     /**
  5766.      * Removes occurrences of specified elements, in specified quantities,
  5767.      * from the specified array. All subsequent elements are shifted left.
  5768.      * For any element-to-be-removed specified in greater quantities than
  5769.      * contained in the original array, no change occurs beyond the
  5770.      * removal of the existing matching items.
  5771.      * <p>
  5772.      * This method returns a new array with the same elements of the input
  5773.      * array except for the earliest-encountered occurrences of the specified
  5774.      * elements. The component type of the returned array is always the same
  5775.      * as that of the input array.
  5776.      * </p>
  5777.      * <pre>
  5778.      * ArrayUtils.removeElements(null, 1, 2)      = null
  5779.      * ArrayUtils.removeElements([], 1, 2)        = []
  5780.      * ArrayUtils.removeElements([1], 2, 3)       = [1]
  5781.      * ArrayUtils.removeElements([1, 3], 1, 2)    = [3]
  5782.      * ArrayUtils.removeElements([1, 3, 1], 1)    = [3, 1]
  5783.      * ArrayUtils.removeElements([1, 3, 1], 1, 1) = [3]
  5784.      * </pre>
  5785.      *
  5786.      * @param array  the array to remove the element from, may be {@code null}
  5787.      * @param values the elements to be removed
  5788.      * @return A new array containing the existing elements except the
  5789.      *         earliest-encountered occurrences of the specified elements.
  5790.      * @since 3.0.1
  5791.      */
  5792.     public static byte[] removeElements(final byte[] array, final byte... values) {
  5793.         if (isEmpty(array) || isEmpty(values)) {
  5794.             return clone(array);
  5795.         }
  5796.         final Map<Byte, MutableInt> occurrences = new HashMap<>(values.length);
  5797.         for (final byte v : values) {
  5798.             final Byte boxed = Byte.valueOf(v);
  5799.             final MutableInt count = occurrences.get(boxed);
  5800.             if (count == null) {
  5801.                 occurrences.put(boxed, new MutableInt(1));
  5802.             } else {
  5803.                 count.increment();
  5804.             }
  5805.         }
  5806.         final BitSet toRemove = new BitSet();
  5807.         for (int i = 0; i < array.length; i++) {
  5808.             final byte key = array[i];
  5809.             final MutableInt count = occurrences.get(key);
  5810.             if (count != null) {
  5811.                 if (count.decrementAndGet() == 0) {
  5812.                     occurrences.remove(key);
  5813.                 }
  5814.                 toRemove.set(i);
  5815.             }
  5816.         }
  5817.         return (byte[]) removeAll(array, toRemove);
  5818.     }

  5819.     /**
  5820.      * Removes occurrences of specified elements, in specified quantities,
  5821.      * from the specified array. All subsequent elements are shifted left.
  5822.      * For any element-to-be-removed specified in greater quantities than
  5823.      * contained in the original array, no change occurs beyond the
  5824.      * removal of the existing matching items.
  5825.      * <p>
  5826.      * This method returns a new array with the same elements of the input
  5827.      * array except for the earliest-encountered occurrences of the specified
  5828.      * elements. The component type of the returned array is always the same
  5829.      * as that of the input array.
  5830.      * </p>
  5831.      * <pre>
  5832.      * ArrayUtils.removeElements(null, 1, 2)      = null
  5833.      * ArrayUtils.removeElements([], 1, 2)        = []
  5834.      * ArrayUtils.removeElements([1], 2, 3)       = [1]
  5835.      * ArrayUtils.removeElements([1, 3], 1, 2)    = [3]
  5836.      * ArrayUtils.removeElements([1, 3, 1], 1)    = [3, 1]
  5837.      * ArrayUtils.removeElements([1, 3, 1], 1, 1) = [3]
  5838.      * </pre>
  5839.      *
  5840.      * @param array  the array to remove the element from, may be {@code null}
  5841.      * @param values the elements to be removed
  5842.      * @return A new array containing the existing elements except the
  5843.      *         earliest-encountered occurrences of the specified elements.
  5844.      * @since 3.0.1
  5845.      */
  5846.     public static char[] removeElements(final char[] array, final char... values) {
  5847.         if (isEmpty(array) || isEmpty(values)) {
  5848.             return clone(array);
  5849.         }
  5850.         final HashMap<Character, MutableInt> occurrences = new HashMap<>(values.length);
  5851.         for (final char v : values) {
  5852.             final Character boxed = Character.valueOf(v);
  5853.             final MutableInt count = occurrences.get(boxed);
  5854.             if (count == null) {
  5855.                 occurrences.put(boxed, new MutableInt(1));
  5856.             } else {
  5857.                 count.increment();
  5858.             }
  5859.         }
  5860.         final BitSet toRemove = new BitSet();
  5861.         for (int i = 0; i < array.length; i++) {
  5862.             final char key = array[i];
  5863.             final MutableInt count = occurrences.get(key);
  5864.             if (count != null) {
  5865.                 if (count.decrementAndGet() == 0) {
  5866.                     occurrences.remove(key);
  5867.                 }
  5868.                 toRemove.set(i);
  5869.             }
  5870.         }
  5871.         return (char[]) removeAll(array, toRemove);
  5872.     }

  5873.     /**
  5874.      * Removes occurrences of specified elements, in specified quantities,
  5875.      * from the specified array. All subsequent elements are shifted left.
  5876.      * For any element-to-be-removed specified in greater quantities than
  5877.      * contained in the original array, no change occurs beyond the
  5878.      * removal of the existing matching items.
  5879.      * <p>
  5880.      * This method returns a new array with the same elements of the input
  5881.      * array except for the earliest-encountered occurrences of the specified
  5882.      * elements. The component type of the returned array is always the same
  5883.      * as that of the input array.
  5884.      * </p>
  5885.      * <pre>
  5886.      * ArrayUtils.removeElements(null, 1, 2)      = null
  5887.      * ArrayUtils.removeElements([], 1, 2)        = []
  5888.      * ArrayUtils.removeElements([1], 2, 3)       = [1]
  5889.      * ArrayUtils.removeElements([1, 3], 1, 2)    = [3]
  5890.      * ArrayUtils.removeElements([1, 3, 1], 1)    = [3, 1]
  5891.      * ArrayUtils.removeElements([1, 3, 1], 1, 1) = [3]
  5892.      * </pre>
  5893.      *
  5894.      * @param array  the array to remove the element from, may be {@code null}
  5895.      * @param values the elements to be removed
  5896.      * @return A new array containing the existing elements except the
  5897.      *         earliest-encountered occurrences of the specified elements.
  5898.      * @since 3.0.1
  5899.      */
  5900.     public static double[] removeElements(final double[] array, final double... values) {
  5901.         if (isEmpty(array) || isEmpty(values)) {
  5902.             return clone(array);
  5903.         }
  5904.         final HashMap<Double, MutableInt> occurrences = new HashMap<>(values.length);
  5905.         for (final double v : values) {
  5906.             final Double boxed = Double.valueOf(v);
  5907.             final MutableInt count = occurrences.get(boxed);
  5908.             if (count == null) {
  5909.                 occurrences.put(boxed, new MutableInt(1));
  5910.             } else {
  5911.                 count.increment();
  5912.             }
  5913.         }
  5914.         final BitSet toRemove = new BitSet();
  5915.         for (int i = 0; i < array.length; i++) {
  5916.             final double key = array[i];
  5917.             final MutableInt count = occurrences.get(key);
  5918.             if (count != null) {
  5919.                 if (count.decrementAndGet() == 0) {
  5920.                     occurrences.remove(key);
  5921.                 }
  5922.                 toRemove.set(i);
  5923.             }
  5924.         }
  5925.         return (double[]) removeAll(array, toRemove);
  5926.     }

  5927.     /**
  5928.      * Removes occurrences of specified elements, in specified quantities,
  5929.      * from the specified array. All subsequent elements are shifted left.
  5930.      * For any element-to-be-removed specified in greater quantities than
  5931.      * contained in the original array, no change occurs beyond the
  5932.      * removal of the existing matching items.
  5933.      * <p>
  5934.      * This method returns a new array with the same elements of the input
  5935.      * array except for the earliest-encountered occurrences of the specified
  5936.      * elements. The component type of the returned array is always the same
  5937.      * as that of the input array.
  5938.      * </p>
  5939.      * <pre>
  5940.      * ArrayUtils.removeElements(null, 1, 2)      = null
  5941.      * ArrayUtils.removeElements([], 1, 2)        = []
  5942.      * ArrayUtils.removeElements([1], 2, 3)       = [1]
  5943.      * ArrayUtils.removeElements([1, 3], 1, 2)    = [3]
  5944.      * ArrayUtils.removeElements([1, 3, 1], 1)    = [3, 1]
  5945.      * ArrayUtils.removeElements([1, 3, 1], 1, 1) = [3]
  5946.      * </pre>
  5947.      *
  5948.      * @param array  the array to remove the element from, may be {@code null}
  5949.      * @param values the elements to be removed
  5950.      * @return A new array containing the existing elements except the
  5951.      *         earliest-encountered occurrences of the specified elements.
  5952.      * @since 3.0.1
  5953.      */
  5954.     public static float[] removeElements(final float[] array, final float... values) {
  5955.         if (isEmpty(array) || isEmpty(values)) {
  5956.             return clone(array);
  5957.         }
  5958.         final HashMap<Float, MutableInt> occurrences = new HashMap<>(values.length);
  5959.         for (final float v : values) {
  5960.             final Float boxed = Float.valueOf(v);
  5961.             final MutableInt count = occurrences.get(boxed);
  5962.             if (count == null) {
  5963.                 occurrences.put(boxed, new MutableInt(1));
  5964.             } else {
  5965.                 count.increment();
  5966.             }
  5967.         }
  5968.         final BitSet toRemove = new BitSet();
  5969.         for (int i = 0; i < array.length; i++) {
  5970.             final float key = array[i];
  5971.             final MutableInt count = occurrences.get(key);
  5972.             if (count != null) {
  5973.                 if (count.decrementAndGet() == 0) {
  5974.                     occurrences.remove(key);
  5975.                 }
  5976.                 toRemove.set(i);
  5977.             }
  5978.         }
  5979.         return (float[]) removeAll(array, toRemove);
  5980.     }

  5981.     /**
  5982.      * Removes occurrences of specified elements, in specified quantities,
  5983.      * from the specified array. All subsequent elements are shifted left.
  5984.      * For any element-to-be-removed specified in greater quantities than
  5985.      * contained in the original array, no change occurs beyond the
  5986.      * removal of the existing matching items.
  5987.      * <p>
  5988.      * This method returns a new array with the same elements of the input
  5989.      * array except for the earliest-encountered occurrences of the specified
  5990.      * elements. The component type of the returned array is always the same
  5991.      * as that of the input array.
  5992.      * </p>
  5993.      * <pre>
  5994.      * ArrayUtils.removeElements(null, 1, 2)      = null
  5995.      * ArrayUtils.removeElements([], 1, 2)        = []
  5996.      * ArrayUtils.removeElements([1], 2, 3)       = [1]
  5997.      * ArrayUtils.removeElements([1, 3], 1, 2)    = [3]
  5998.      * ArrayUtils.removeElements([1, 3, 1], 1)    = [3, 1]
  5999.      * ArrayUtils.removeElements([1, 3, 1], 1, 1) = [3]
  6000.      * </pre>
  6001.      *
  6002.      * @param array  the array to remove the element from, may be {@code null}
  6003.      * @param values the elements to be removed
  6004.      * @return A new array containing the existing elements except the
  6005.      *         earliest-encountered occurrences of the specified elements.
  6006.      * @since 3.0.1
  6007.      */
  6008.     public static int[] removeElements(final int[] array, final int... values) {
  6009.         if (isEmpty(array) || isEmpty(values)) {
  6010.             return clone(array);
  6011.         }
  6012.         final HashMap<Integer, MutableInt> occurrences = new HashMap<>(values.length);
  6013.         for (final int v : values) {
  6014.             final Integer boxed = Integer.valueOf(v);
  6015.             final MutableInt count = occurrences.get(boxed);
  6016.             if (count == null) {
  6017.                 occurrences.put(boxed, new MutableInt(1));
  6018.             } else {
  6019.                 count.increment();
  6020.             }
  6021.         }
  6022.         final BitSet toRemove = new BitSet();
  6023.         for (int i = 0; i < array.length; i++) {
  6024.             final int key = array[i];
  6025.             final MutableInt count = occurrences.get(key);
  6026.             if (count != null) {
  6027.                 if (count.decrementAndGet() == 0) {
  6028.                     occurrences.remove(key);
  6029.                 }
  6030.                 toRemove.set(i);
  6031.             }
  6032.         }
  6033.         return (int[]) removeAll(array, toRemove);
  6034.     }

  6035.     /**
  6036.      * Removes occurrences of specified elements, in specified quantities,
  6037.      * from the specified array. All subsequent elements are shifted left.
  6038.      * For any element-to-be-removed specified in greater quantities than
  6039.      * contained in the original array, no change occurs beyond the
  6040.      * removal of the existing matching items.
  6041.      * <p>
  6042.      * This method returns a new array with the same elements of the input
  6043.      * array except for the earliest-encountered occurrences of the specified
  6044.      * elements. The component type of the returned array is always the same
  6045.      * as that of the input array.
  6046.      * </p>
  6047.      * <pre>
  6048.      * ArrayUtils.removeElements(null, 1, 2)      = null
  6049.      * ArrayUtils.removeElements([], 1, 2)        = []
  6050.      * ArrayUtils.removeElements([1], 2, 3)       = [1]
  6051.      * ArrayUtils.removeElements([1, 3], 1, 2)    = [3]
  6052.      * ArrayUtils.removeElements([1, 3, 1], 1)    = [3, 1]
  6053.      * ArrayUtils.removeElements([1, 3, 1], 1, 1) = [3]
  6054.      * </pre>
  6055.      *
  6056.      * @param array  the array to remove the element from, may be {@code null}
  6057.      * @param values the elements to be removed
  6058.      * @return A new array containing the existing elements except the
  6059.      *         earliest-encountered occurrences of the specified elements.
  6060.      * @since 3.0.1
  6061.      */
  6062.     public static long[] removeElements(final long[] array, final long... values) {
  6063.         if (isEmpty(array) || isEmpty(values)) {
  6064.             return clone(array);
  6065.         }
  6066.         final HashMap<Long, MutableInt> occurrences = new HashMap<>(values.length);
  6067.         for (final long v : values) {
  6068.             final Long boxed = Long.valueOf(v);
  6069.             final MutableInt count = occurrences.get(boxed);
  6070.             if (count == null) {
  6071.                 occurrences.put(boxed, new MutableInt(1));
  6072.             } else {
  6073.                 count.increment();
  6074.             }
  6075.         }
  6076.         final BitSet toRemove = new BitSet();
  6077.         for (int i = 0; i < array.length; i++) {
  6078.             final long key = array[i];
  6079.             final MutableInt count = occurrences.get(key);
  6080.             if (count != null) {
  6081.                 if (count.decrementAndGet() == 0) {
  6082.                     occurrences.remove(key);
  6083.                 }
  6084.                 toRemove.set(i);
  6085.             }
  6086.         }
  6087.         return (long[]) removeAll(array, toRemove);
  6088.     }

  6089.     /**
  6090.      * Removes occurrences of specified elements, in specified quantities,
  6091.      * from the specified array. All subsequent elements are shifted left.
  6092.      * For any element-to-be-removed specified in greater quantities than
  6093.      * contained in the original array, no change occurs beyond the
  6094.      * removal of the existing matching items.
  6095.      * <p>
  6096.      * This method returns a new array with the same elements of the input
  6097.      * array except for the earliest-encountered occurrences of the specified
  6098.      * elements. The component type of the returned array is always the same
  6099.      * as that of the input array.
  6100.      * </p>
  6101.      * <pre>
  6102.      * ArrayUtils.removeElements(null, 1, 2)      = null
  6103.      * ArrayUtils.removeElements([], 1, 2)        = []
  6104.      * ArrayUtils.removeElements([1], 2, 3)       = [1]
  6105.      * ArrayUtils.removeElements([1, 3], 1, 2)    = [3]
  6106.      * ArrayUtils.removeElements([1, 3, 1], 1)    = [3, 1]
  6107.      * ArrayUtils.removeElements([1, 3, 1], 1, 1) = [3]
  6108.      * </pre>
  6109.      *
  6110.      * @param array  the array to remove the element from, may be {@code null}
  6111.      * @param values the elements to be removed
  6112.      * @return A new array containing the existing elements except the
  6113.      *         earliest-encountered occurrences of the specified elements.
  6114.      * @since 3.0.1
  6115.      */
  6116.     public static short[] removeElements(final short[] array, final short... values) {
  6117.         if (isEmpty(array) || isEmpty(values)) {
  6118.             return clone(array);
  6119.         }
  6120.         final HashMap<Short, MutableInt> occurrences = new HashMap<>(values.length);
  6121.         for (final short v : values) {
  6122.             final Short boxed = Short.valueOf(v);
  6123.             final MutableInt count = occurrences.get(boxed);
  6124.             if (count == null) {
  6125.                 occurrences.put(boxed, new MutableInt(1));
  6126.             } else {
  6127.                 count.increment();
  6128.             }
  6129.         }
  6130.         final BitSet toRemove = new BitSet();
  6131.         for (int i = 0; i < array.length; i++) {
  6132.             final short key = array[i];
  6133.             final MutableInt count = occurrences.get(key);
  6134.             if (count != null) {
  6135.                 if (count.decrementAndGet() == 0) {
  6136.                     occurrences.remove(key);
  6137.                 }
  6138.                 toRemove.set(i);
  6139.             }
  6140.         }
  6141.         return (short[]) removeAll(array, toRemove);
  6142.     }

  6143.     /**
  6144.      * Removes occurrences of specified elements, in specified quantities,
  6145.      * from the specified array. All subsequent elements are shifted left.
  6146.      * For any element-to-be-removed specified in greater quantities than
  6147.      * contained in the original array, no change occurs beyond the
  6148.      * removal of the existing matching items.
  6149.      * <p>
  6150.      * This method returns a new array with the same elements of the input
  6151.      * array except for the earliest-encountered occurrences of the specified
  6152.      * elements. The component type of the returned array is always the same
  6153.      * as that of the input array.
  6154.      * </p>
  6155.      * <pre>
  6156.      * ArrayUtils.removeElements(null, "a", "b")            = null
  6157.      * ArrayUtils.removeElements([], "a", "b")              = []
  6158.      * ArrayUtils.removeElements(["a"], "b", "c")           = ["a"]
  6159.      * ArrayUtils.removeElements(["a", "b"], "a", "c")      = ["b"]
  6160.      * ArrayUtils.removeElements(["a", "b", "a"], "a")      = ["b", "a"]
  6161.      * ArrayUtils.removeElements(["a", "b", "a"], "a", "a") = ["b"]
  6162.      * </pre>
  6163.      *
  6164.      * @param <T> the component type of the array
  6165.      * @param array  the array to remove the element from, may be {@code null}
  6166.      * @param values the elements to be removed
  6167.      * @return A new array containing the existing elements except the
  6168.      *         earliest-encountered occurrences of the specified elements.
  6169.      * @since 3.0.1
  6170.      */
  6171.     @SafeVarargs
  6172.     public static <T> T[] removeElements(final T[] array, final T... values) {
  6173.         if (isEmpty(array) || isEmpty(values)) {
  6174.             return clone(array);
  6175.         }
  6176.         final HashMap<T, MutableInt> occurrences = new HashMap<>(values.length);
  6177.         for (final T v : values) {
  6178.             final MutableInt count = occurrences.get(v);
  6179.             if (count == null) {
  6180.                 occurrences.put(v, new MutableInt(1));
  6181.             } else {
  6182.                 count.increment();
  6183.             }
  6184.         }
  6185.         final BitSet toRemove = new BitSet();
  6186.         for (int i = 0; i < array.length; i++) {
  6187.             final T key = array[i];
  6188.             final MutableInt count = occurrences.get(key);
  6189.             if (count != null) {
  6190.                 if (count.decrementAndGet() == 0) {
  6191.                     occurrences.remove(key);
  6192.                 }
  6193.                 toRemove.set(i);
  6194.             }
  6195.         }
  6196.         @SuppressWarnings("unchecked") // removeAll() always creates an array of the same type as its input
  6197.         final T[] result = (T[]) removeAll(array, toRemove);
  6198.         return result;
  6199.     }

  6200.     /**
  6201.      * Reverses the order of the given array.
  6202.      * <p>
  6203.      * This method does nothing for a {@code null} input array.
  6204.      * </p>
  6205.      *
  6206.      * @param array  the array to reverse, may be {@code null}
  6207.      */
  6208.     public static void reverse(final boolean[] array) {
  6209.         if (array == null) {
  6210.             return;
  6211.         }
  6212.         reverse(array, 0, array.length);
  6213.     }

  6214.     /**
  6215.      * Reverses the order of the given array in the given range.
  6216.      * <p>
  6217.      * This method does nothing for a {@code null} input array.
  6218.      * </p>
  6219.      *
  6220.      * @param array
  6221.      *            the array to reverse, may be {@code null}
  6222.      * @param startIndexInclusive
  6223.      *            the starting index. Undervalue (&lt;0) is promoted to 0, overvalue (&gt;array.length) results in no
  6224.      *            change.
  6225.      * @param endIndexExclusive
  6226.      *            elements up to endIndex-1 are reversed in the array. Undervalue (&lt; start index) results in no
  6227.      *            change. Overvalue (&gt;array.length) is demoted to array length.
  6228.      * @since 3.2
  6229.      */
  6230.     public static void reverse(final boolean[] array, final int startIndexInclusive, final int endIndexExclusive) {
  6231.         if (array == null) {
  6232.             return;
  6233.         }
  6234.         int i = Math.max(startIndexInclusive, 0);
  6235.         int j = Math.min(array.length, endIndexExclusive) - 1;
  6236.         boolean tmp;
  6237.         while (j > i) {
  6238.             tmp = array[j];
  6239.             array[j] = array[i];
  6240.             array[i] = tmp;
  6241.             j--;
  6242.             i++;
  6243.         }
  6244.     }

  6245.     /**
  6246.      * Reverses the order of the given array.
  6247.      * <p>
  6248.      * This method does nothing for a {@code null} input array.
  6249.      * </p>
  6250.      *
  6251.      * @param array  the array to reverse, may be {@code null}
  6252.      */
  6253.     public static void reverse(final byte[] array) {
  6254.         if (array != null) {
  6255.             reverse(array, 0, array.length);
  6256.         }
  6257.     }

  6258.     /**
  6259.      * Reverses the order of the given array in the given range.
  6260.      * <p>
  6261.      * This method does nothing for a {@code null} input array.
  6262.      * </p>
  6263.      *
  6264.      * @param array
  6265.      *            the array to reverse, may be {@code null}
  6266.      * @param startIndexInclusive
  6267.      *            the starting index. Undervalue (&lt;0) is promoted to 0, overvalue (&gt;array.length) results in no
  6268.      *            change.
  6269.      * @param endIndexExclusive
  6270.      *            elements up to endIndex-1 are reversed in the array. Undervalue (&lt; start index) results in no
  6271.      *            change. Overvalue (&gt;array.length) is demoted to array length.
  6272.      * @since 3.2
  6273.      */
  6274.     public static void reverse(final byte[] array, final int startIndexInclusive, final int endIndexExclusive) {
  6275.         if (array == null) {
  6276.             return;
  6277.         }
  6278.         int i = Math.max(startIndexInclusive, 0);
  6279.         int j = Math.min(array.length, endIndexExclusive) - 1;
  6280.         byte tmp;
  6281.         while (j > i) {
  6282.             tmp = array[j];
  6283.             array[j] = array[i];
  6284.             array[i] = tmp;
  6285.             j--;
  6286.             i++;
  6287.         }
  6288.     }

  6289.     /**
  6290.      * Reverses the order of the given array.
  6291.      * <p>
  6292.      * This method does nothing for a {@code null} input array.
  6293.      * </p>
  6294.      *
  6295.      * @param array  the array to reverse, may be {@code null}
  6296.      */
  6297.     public static void reverse(final char[] array) {
  6298.         if (array != null) {
  6299.             reverse(array, 0, array.length);
  6300.         }
  6301.     }

  6302.     /**
  6303.      * Reverses the order of the given array in the given range.
  6304.      * <p>
  6305.      * This method does nothing for a {@code null} input array.
  6306.      * </p>
  6307.      *
  6308.      * @param array
  6309.      *            the array to reverse, may be {@code null}
  6310.      * @param startIndexInclusive
  6311.      *            the starting index. Undervalue (&lt;0) is promoted to 0, overvalue (&gt;array.length) results in no
  6312.      *            change.
  6313.      * @param endIndexExclusive
  6314.      *            elements up to endIndex-1 are reversed in the array. Undervalue (&lt; start index) results in no
  6315.      *            change. Overvalue (&gt;array.length) is demoted to array length.
  6316.      * @since 3.2
  6317.      */
  6318.     public static void reverse(final char[] array, final int startIndexInclusive, final int endIndexExclusive) {
  6319.         if (array == null) {
  6320.             return;
  6321.         }
  6322.         int i = Math.max(startIndexInclusive, 0);
  6323.         int j = Math.min(array.length, endIndexExclusive) - 1;
  6324.         char tmp;
  6325.         while (j > i) {
  6326.             tmp = array[j];
  6327.             array[j] = array[i];
  6328.             array[i] = tmp;
  6329.             j--;
  6330.             i++;
  6331.         }
  6332.     }

  6333.     /**
  6334.      * Reverses the order of the given array.
  6335.      * <p>
  6336.      * This method does nothing for a {@code null} input array.
  6337.      * </p>
  6338.      *
  6339.      * @param array  the array to reverse, may be {@code null}
  6340.      */
  6341.     public static void reverse(final double[] array) {
  6342.         if (array != null) {
  6343.             reverse(array, 0, array.length);
  6344.         }
  6345.     }

  6346.     /**
  6347.      * Reverses the order of the given array in the given range.
  6348.      * <p>
  6349.      * This method does nothing for a {@code null} input array.
  6350.      * </p>
  6351.      *
  6352.      * @param array
  6353.      *            the array to reverse, may be {@code null}
  6354.      * @param startIndexInclusive
  6355.      *            the starting index. Undervalue (&lt;0) is promoted to 0, overvalue (&gt;array.length) results in no
  6356.      *            change.
  6357.      * @param endIndexExclusive
  6358.      *            elements up to endIndex-1 are reversed in the array. Undervalue (&lt; start index) results in no
  6359.      *            change. Overvalue (&gt;array.length) is demoted to array length.
  6360.      * @since 3.2
  6361.      */
  6362.     public static void reverse(final double[] array, final int startIndexInclusive, final int endIndexExclusive) {
  6363.         if (array == null) {
  6364.             return;
  6365.         }
  6366.         int i = Math.max(startIndexInclusive, 0);
  6367.         int j = Math.min(array.length, endIndexExclusive) - 1;
  6368.         double tmp;
  6369.         while (j > i) {
  6370.             tmp = array[j];
  6371.             array[j] = array[i];
  6372.             array[i] = tmp;
  6373.             j--;
  6374.             i++;
  6375.         }
  6376.     }

  6377.     /**
  6378.      * Reverses the order of the given array.
  6379.      * <p>
  6380.      * This method does nothing for a {@code null} input array.
  6381.      * </p>
  6382.      *
  6383.      * @param array  the array to reverse, may be {@code null}
  6384.      */
  6385.     public static void reverse(final float[] array) {
  6386.         if (array != null) {
  6387.             reverse(array, 0, array.length);
  6388.         }
  6389.     }

  6390.     /**
  6391.      * Reverses the order of the given array in the given range.
  6392.      * <p>
  6393.      * This method does nothing for a {@code null} input array.
  6394.      * </p>
  6395.      *
  6396.      * @param array
  6397.      *            the array to reverse, may be {@code null}
  6398.      * @param startIndexInclusive
  6399.      *            the starting index. Undervalue (&lt;0) is promoted to 0, overvalue (&gt;array.length) results in no
  6400.      *            change.
  6401.      * @param endIndexExclusive
  6402.      *            elements up to endIndex-1 are reversed in the array. Undervalue (&lt; start index) results in no
  6403.      *            change. Overvalue (&gt;array.length) is demoted to array length.
  6404.      * @since 3.2
  6405.      */
  6406.     public static void reverse(final float[] array, final int startIndexInclusive, final int endIndexExclusive) {
  6407.         if (array == null) {
  6408.             return;
  6409.         }
  6410.         int i = Math.max(startIndexInclusive, 0);
  6411.         int j = Math.min(array.length, endIndexExclusive) - 1;
  6412.         float tmp;
  6413.         while (j > i) {
  6414.             tmp = array[j];
  6415.             array[j] = array[i];
  6416.             array[i] = tmp;
  6417.             j--;
  6418.             i++;
  6419.         }
  6420.     }

  6421.     /**
  6422.      * Reverses the order of the given array.
  6423.      * <p>
  6424.      * This method does nothing for a {@code null} input array.
  6425.      * </p>
  6426.      *
  6427.      * @param array  the array to reverse, may be {@code null}
  6428.      */
  6429.     public static void reverse(final int[] array) {
  6430.         if (array != null) {
  6431.             reverse(array, 0, array.length);
  6432.         }
  6433.     }

  6434.     /**
  6435.      * Reverses the order of the given array in the given range.
  6436.      * <p>
  6437.      * This method does nothing for a {@code null} input array.
  6438.      * </p>
  6439.      *
  6440.      * @param array
  6441.      *            the array to reverse, may be {@code null}
  6442.      * @param startIndexInclusive
  6443.      *            the starting index. Undervalue (&lt;0) is promoted to 0, overvalue (&gt;array.length) results in no
  6444.      *            change.
  6445.      * @param endIndexExclusive
  6446.      *            elements up to endIndex-1 are reversed in the array. Undervalue (&lt; start index) results in no
  6447.      *            change. Overvalue (&gt;array.length) is demoted to array length.
  6448.      * @since 3.2
  6449.      */
  6450.     public static void reverse(final int[] array, final int startIndexInclusive, final int endIndexExclusive) {
  6451.         if (array == null) {
  6452.             return;
  6453.         }
  6454.         int i = Math.max(startIndexInclusive, 0);
  6455.         int j = Math.min(array.length, endIndexExclusive) - 1;
  6456.         int tmp;
  6457.         while (j > i) {
  6458.             tmp = array[j];
  6459.             array[j] = array[i];
  6460.             array[i] = tmp;
  6461.             j--;
  6462.             i++;
  6463.         }
  6464.     }

  6465.     /**
  6466.      * Reverses the order of the given array.
  6467.      * <p>
  6468.      * This method does nothing for a {@code null} input array.
  6469.      * </p>
  6470.      *
  6471.      * @param array  the array to reverse, may be {@code null}
  6472.      */
  6473.     public static void reverse(final long[] array) {
  6474.         if (array != null) {
  6475.             reverse(array, 0, array.length);
  6476.         }
  6477.     }

  6478.     /**
  6479.      * Reverses the order of the given array in the given range.
  6480.      * <p>
  6481.      * This method does nothing for a {@code null} input array.
  6482.      * </p>
  6483.      *
  6484.      * @param array
  6485.      *            the array to reverse, may be {@code null}
  6486.      * @param startIndexInclusive
  6487.      *            the starting index. Undervalue (&lt;0) is promoted to 0, overvalue (&gt;array.length) results in no
  6488.      *            change.
  6489.      * @param endIndexExclusive
  6490.      *            elements up to endIndex-1 are reversed in the array. Undervalue (&lt; start index) results in no
  6491.      *            change. Overvalue (&gt;array.length) is demoted to array length.
  6492.      * @since 3.2
  6493.      */
  6494.     public static void reverse(final long[] array, final int startIndexInclusive, final int endIndexExclusive) {
  6495.         if (array == null) {
  6496.             return;
  6497.         }
  6498.         int i = Math.max(startIndexInclusive, 0);
  6499.         int j = Math.min(array.length, endIndexExclusive) - 1;
  6500.         long tmp;
  6501.         while (j > i) {
  6502.             tmp = array[j];
  6503.             array[j] = array[i];
  6504.             array[i] = tmp;
  6505.             j--;
  6506.             i++;
  6507.         }
  6508.     }

  6509.     /**
  6510.      * Reverses the order of the given array.
  6511.      * <p>
  6512.      * There is no special handling for multi-dimensional arrays.
  6513.      * </p>
  6514.      * <p>
  6515.      * This method does nothing for a {@code null} input array.
  6516.      * </p>
  6517.      *
  6518.      * @param array  the array to reverse, may be {@code null}
  6519.      */
  6520.     public static void reverse(final Object[] array) {
  6521.         if (array != null) {
  6522.             reverse(array, 0, array.length);
  6523.         }
  6524.     }

  6525.     /**
  6526.      * Reverses the order of the given array in the given range.
  6527.      * <p>
  6528.      * This method does nothing for a {@code null} input array.
  6529.      * </p>
  6530.      *
  6531.      * @param array
  6532.      *            the array to reverse, may be {@code null}
  6533.      * @param startIndexInclusive
  6534.      *            the starting index. Under value (&lt;0) is promoted to 0, over value (&gt;array.length) results in no
  6535.      *            change.
  6536.      * @param endIndexExclusive
  6537.      *            elements up to endIndex-1 are reversed in the array. Under value (&lt; start index) results in no
  6538.      *            change. Over value (&gt;array.length) is demoted to array length.
  6539.      * @since 3.2
  6540.      */
  6541.     public static void reverse(final Object[] array, final int startIndexInclusive, final int endIndexExclusive) {
  6542.         if (array == null) {
  6543.             return;
  6544.         }
  6545.         int i = Math.max(startIndexInclusive, 0);
  6546.         int j = Math.min(array.length, endIndexExclusive) - 1;
  6547.         Object tmp;
  6548.         while (j > i) {
  6549.             tmp = array[j];
  6550.             array[j] = array[i];
  6551.             array[i] = tmp;
  6552.             j--;
  6553.             i++;
  6554.         }
  6555.     }

  6556.     /**
  6557.      * Reverses the order of the given array.
  6558.      * <p>
  6559.      * This method does nothing for a {@code null} input array.
  6560.      * </p>
  6561.      *
  6562.      * @param array  the array to reverse, may be {@code null}
  6563.      */
  6564.     public static void reverse(final short[] array) {
  6565.         if (array != null) {
  6566.             reverse(array, 0, array.length);
  6567.         }
  6568.     }

  6569.     /**
  6570.      * Reverses the order of the given array in the given range.
  6571.      * <p>
  6572.      * This method does nothing for a {@code null} input array.
  6573.      * </p>
  6574.      *
  6575.      * @param array
  6576.      *            the array to reverse, may be {@code null}
  6577.      * @param startIndexInclusive
  6578.      *            the starting index. Undervalue (&lt;0) is promoted to 0, overvalue (&gt;array.length) results in no
  6579.      *            change.
  6580.      * @param endIndexExclusive
  6581.      *            elements up to endIndex-1 are reversed in the array. Undervalue (&lt; start index) results in no
  6582.      *            change. Overvalue (&gt;array.length) is demoted to array length.
  6583.      * @since 3.2
  6584.      */
  6585.     public static void reverse(final short[] array, final int startIndexInclusive, final int endIndexExclusive) {
  6586.         if (array == null) {
  6587.             return;
  6588.         }
  6589.         int i = Math.max(startIndexInclusive, 0);
  6590.         int j = Math.min(array.length, endIndexExclusive) - 1;
  6591.         short tmp;
  6592.         while (j > i) {
  6593.             tmp = array[j];
  6594.             array[j] = array[i];
  6595.             array[i] = tmp;
  6596.             j--;
  6597.             i++;
  6598.         }
  6599.     }

  6600.     /**
  6601.      * Sets all elements of the specified array, using the provided generator supplier to compute each element.
  6602.      * <p>
  6603.      * If the generator supplier throws an exception, it is relayed to the caller and the array is left in an indeterminate
  6604.      * state.
  6605.      * </p>
  6606.      *
  6607.      * @param <T> type of elements of the array.
  6608.      * @param array array to be initialized.
  6609.      * @param generator a function accepting an index and producing the desired value for that position.
  6610.      * @return the input array
  6611.      * @since 3.13.0
  6612.      */
  6613.     public static <T> T[] setAll(final T[] array, final IntFunction<? extends T> generator) {
  6614.         if (array != null && generator != null) {
  6615.             Arrays.setAll(array, generator);
  6616.         }
  6617.         return array;
  6618.     }

  6619.     /**
  6620.      * Sets all elements of the specified array, using the provided generator supplier to compute each element.
  6621.      * <p>
  6622.      * If the generator supplier throws an exception, it is relayed to the caller and the array is left in an indeterminate
  6623.      * state.
  6624.      * </p>
  6625.      *
  6626.      * @param <T> type of elements of the array.
  6627.      * @param array array to be initialized.
  6628.      * @param generator a function accepting an index and producing the desired value for that position.
  6629.      * @return the input array
  6630.      * @since 3.13.0
  6631.      */
  6632.     public static <T> T[] setAll(final T[] array, final Supplier<? extends T> generator) {
  6633.         if (array != null && generator != null) {
  6634.             for (int i = 0; i < array.length; i++) {
  6635.                 array[i] = generator.get();
  6636.             }
  6637.         }
  6638.         return array;
  6639.     }

  6640.     /**
  6641.      * Shifts the order of the given boolean array.
  6642.      *
  6643.      * <p>There is no special handling for multi-dimensional arrays. This method
  6644.      * does nothing for {@code null} or empty input arrays.</p>
  6645.      *
  6646.      * @param array  the array to shift, may be {@code null}
  6647.      * @param offset
  6648.      *          The number of positions to rotate the elements.  If the offset is larger than the number of elements to
  6649.      *          rotate, than the effective offset is modulo the number of elements to rotate.
  6650.      * @since 3.5
  6651.      */
  6652.     public static void shift(final boolean[] array, final int offset) {
  6653.         if (array != null) {
  6654.             shift(array, 0, array.length, offset);
  6655.         }
  6656.     }

  6657.     /**
  6658.      * Shifts the order of a series of elements in the given boolean array.
  6659.      *
  6660.      * <p>There is no special handling for multi-dimensional arrays. This method
  6661.      * does nothing for {@code null} or empty input arrays.</p>
  6662.      *
  6663.      * @param array
  6664.      *            the array to shift, may be {@code null}
  6665.      * @param startIndexInclusive
  6666.      *            the starting index. Undervalue (&lt;0) is promoted to 0, overvalue (&gt;array.length) results in no
  6667.      *            change.
  6668.      * @param endIndexExclusive
  6669.      *            elements up to endIndex-1 are shifted in the array. Undervalue (&lt; start index) results in no
  6670.      *            change. Overvalue (&gt;array.length) is demoted to array length.
  6671.      * @param offset
  6672.      *          The number of positions to rotate the elements.  If the offset is larger than the number of elements to
  6673.      *          rotate, than the effective offset is modulo the number of elements to rotate.
  6674.      * @since 3.5
  6675.      */
  6676.     public static void shift(final boolean[] array, int startIndexInclusive, int endIndexExclusive, int offset) {
  6677.         if (array == null || startIndexInclusive >= array.length - 1 || endIndexExclusive <= 0) {
  6678.             return;
  6679.         }
  6680.         startIndexInclusive = max0(startIndexInclusive);
  6681.         endIndexExclusive = Math.min(endIndexExclusive, array.length);
  6682.         int n = endIndexExclusive - startIndexInclusive;
  6683.         if (n <= 1) {
  6684.             return;
  6685.         }
  6686.         offset %= n;
  6687.         if (offset < 0) {
  6688.             offset += n;
  6689.         }
  6690.         // For algorithm explanations and proof of O(n) time complexity and O(1) space complexity
  6691.         // see https://beradrian.wordpress.com/2015/04/07/shift-an-array-in-on-in-place/
  6692.         while (n > 1 && offset > 0) {
  6693.             final int nOffset = n - offset;
  6694.             if (offset > nOffset) {
  6695.                 swap(array, startIndexInclusive, startIndexInclusive + n - nOffset,  nOffset);
  6696.                 n = offset;
  6697.                 offset -= nOffset;
  6698.             } else if (offset < nOffset) {
  6699.                 swap(array, startIndexInclusive, startIndexInclusive + nOffset,  offset);
  6700.                 startIndexInclusive += offset;
  6701.                 n = nOffset;
  6702.             } else {
  6703.                 swap(array, startIndexInclusive, startIndexInclusive + nOffset, offset);
  6704.                 break;
  6705.             }
  6706.         }
  6707.     }

  6708.     /**
  6709.      * Shifts the order of the given byte array.
  6710.      *
  6711.      * <p>There is no special handling for multi-dimensional arrays. This method
  6712.      * does nothing for {@code null} or empty input arrays.</p>
  6713.      *
  6714.      * @param array  the array to shift, may be {@code null}
  6715.      * @param offset
  6716.      *          The number of positions to rotate the elements.  If the offset is larger than the number of elements to
  6717.      *          rotate, than the effective offset is modulo the number of elements to rotate.
  6718.      * @since 3.5
  6719.      */
  6720.     public static void shift(final byte[] array, final int offset) {
  6721.         if (array != null) {
  6722.             shift(array, 0, array.length, offset);
  6723.         }
  6724.     }

  6725.     /**
  6726.      * Shifts the order of a series of elements in the given byte array.
  6727.      *
  6728.      * <p>There is no special handling for multi-dimensional arrays. This method
  6729.      * does nothing for {@code null} or empty input arrays.</p>
  6730.      *
  6731.      * @param array
  6732.      *            the array to shift, may be {@code null}
  6733.      * @param startIndexInclusive
  6734.      *            the starting index. Undervalue (&lt;0) is promoted to 0, overvalue (&gt;array.length) results in no
  6735.      *            change.
  6736.      * @param endIndexExclusive
  6737.      *            elements up to endIndex-1 are shifted in the array. Undervalue (&lt; start index) results in no
  6738.      *            change. Overvalue (&gt;array.length) is demoted to array length.
  6739.      * @param offset
  6740.      *          The number of positions to rotate the elements.  If the offset is larger than the number of elements to
  6741.      *          rotate, than the effective offset is modulo the number of elements to rotate.
  6742.      * @since 3.5
  6743.      */
  6744.     public static void shift(final byte[] array, int startIndexInclusive, int endIndexExclusive, int offset) {
  6745.         if (array == null || startIndexInclusive >= array.length - 1 || endIndexExclusive <= 0) {
  6746.             return;
  6747.         }
  6748.         startIndexInclusive = max0(startIndexInclusive);
  6749.         endIndexExclusive = Math.min(endIndexExclusive, array.length);
  6750.         int n = endIndexExclusive - startIndexInclusive;
  6751.         if (n <= 1) {
  6752.             return;
  6753.         }
  6754.         offset %= n;
  6755.         if (offset < 0) {
  6756.             offset += n;
  6757.         }
  6758.         // For algorithm explanations and proof of O(n) time complexity and O(1) space complexity
  6759.         // see https://beradrian.wordpress.com/2015/04/07/shift-an-array-in-on-in-place/
  6760.         while (n > 1 && offset > 0) {
  6761.             final int nOffset = n - offset;
  6762.             if (offset > nOffset) {
  6763.                 swap(array, startIndexInclusive, startIndexInclusive + n - nOffset,  nOffset);
  6764.                 n = offset;
  6765.                 offset -= nOffset;
  6766.             } else if (offset < nOffset) {
  6767.                 swap(array, startIndexInclusive, startIndexInclusive + nOffset,  offset);
  6768.                 startIndexInclusive += offset;
  6769.                 n = nOffset;
  6770.             } else {
  6771.                 swap(array, startIndexInclusive, startIndexInclusive + nOffset, offset);
  6772.                 break;
  6773.             }
  6774.         }
  6775.     }

  6776.     /**
  6777.      * Shifts the order of the given char array.
  6778.      *
  6779.      * <p>There is no special handling for multi-dimensional arrays. This method
  6780.      * does nothing for {@code null} or empty input arrays.</p>
  6781.      *
  6782.      * @param array  the array to shift, may be {@code null}
  6783.      * @param offset
  6784.      *          The number of positions to rotate the elements.  If the offset is larger than the number of elements to
  6785.      *          rotate, than the effective offset is modulo the number of elements to rotate.
  6786.      * @since 3.5
  6787.      */
  6788.     public static void shift(final char[] array, final int offset) {
  6789.         if (array != null) {
  6790.             shift(array, 0, array.length, offset);
  6791.         }
  6792.     }

  6793.     /**
  6794.      * Shifts the order of a series of elements in the given char array.
  6795.      *
  6796.      * <p>There is no special handling for multi-dimensional arrays. This method
  6797.      * does nothing for {@code null} or empty input arrays.</p>
  6798.      *
  6799.      * @param array
  6800.      *            the array to shift, may be {@code null}
  6801.      * @param startIndexInclusive
  6802.      *            the starting index. Undervalue (&lt;0) is promoted to 0, overvalue (&gt;array.length) results in no
  6803.      *            change.
  6804.      * @param endIndexExclusive
  6805.      *            elements up to endIndex-1 are shifted in the array. Undervalue (&lt; start index) results in no
  6806.      *            change. Overvalue (&gt;array.length) is demoted to array length.
  6807.      * @param offset
  6808.      *          The number of positions to rotate the elements.  If the offset is larger than the number of elements to
  6809.      *          rotate, than the effective offset is modulo the number of elements to rotate.
  6810.      * @since 3.5
  6811.      */
  6812.     public static void shift(final char[] array, int startIndexInclusive, int endIndexExclusive, int offset) {
  6813.         if (array == null || startIndexInclusive >= array.length - 1 || endIndexExclusive <= 0) {
  6814.             return;
  6815.         }
  6816.         startIndexInclusive = max0(startIndexInclusive);
  6817.         endIndexExclusive = Math.min(endIndexExclusive, array.length);
  6818.         int n = endIndexExclusive - startIndexInclusive;
  6819.         if (n <= 1) {
  6820.             return;
  6821.         }
  6822.         offset %= n;
  6823.         if (offset < 0) {
  6824.             offset += n;
  6825.         }
  6826.         // For algorithm explanations and proof of O(n) time complexity and O(1) space complexity
  6827.         // see https://beradrian.wordpress.com/2015/04/07/shift-an-array-in-on-in-place/
  6828.         while (n > 1 && offset > 0) {
  6829.             final int nOffset = n - offset;
  6830.             if (offset > nOffset) {
  6831.                 swap(array, startIndexInclusive, startIndexInclusive + n - nOffset,  nOffset);
  6832.                 n = offset;
  6833.                 offset -= nOffset;
  6834.             } else if (offset < nOffset) {
  6835.                 swap(array, startIndexInclusive, startIndexInclusive + nOffset,  offset);
  6836.                 startIndexInclusive += offset;
  6837.                 n = nOffset;
  6838.             } else {
  6839.                 swap(array, startIndexInclusive, startIndexInclusive + nOffset, offset);
  6840.                 break;
  6841.             }
  6842.         }
  6843.     }

  6844.     /**
  6845.      * Shifts the order of the given double array.
  6846.      *
  6847.      * <p>There is no special handling for multi-dimensional arrays. This method
  6848.      * does nothing for {@code null} or empty input arrays.</p>
  6849.      *
  6850.      * @param array  the array to shift, may be {@code null}
  6851.      * @param offset
  6852.      *          The number of positions to rotate the elements.  If the offset is larger than the number of elements to
  6853.      *          rotate, than the effective offset is modulo the number of elements to rotate.
  6854.      * @since 3.5
  6855.      */
  6856.     public static void shift(final double[] array, final int offset) {
  6857.         if (array != null) {
  6858.             shift(array, 0, array.length, offset);
  6859.         }
  6860.     }

  6861.     /**
  6862.      * Shifts the order of a series of elements in the given double array.
  6863.      *
  6864.      * <p>There is no special handling for multi-dimensional arrays. This method
  6865.      * does nothing for {@code null} or empty input arrays.</p>
  6866.      *
  6867.      * @param array
  6868.      *            the array to shift, may be {@code null}
  6869.      * @param startIndexInclusive
  6870.      *            the starting index. Undervalue (&lt;0) is promoted to 0, overvalue (&gt;array.length) results in no
  6871.      *            change.
  6872.      * @param endIndexExclusive
  6873.      *            elements up to endIndex-1 are shifted in the array. Undervalue (&lt; start index) results in no
  6874.      *            change. Overvalue (&gt;array.length) is demoted to array length.
  6875.      * @param offset
  6876.      *          The number of positions to rotate the elements.  If the offset is larger than the number of elements to
  6877.      *          rotate, than the effective offset is modulo the number of elements to rotate.
  6878.      * @since 3.5
  6879.      */
  6880.     public static void shift(final double[] array, int startIndexInclusive, int endIndexExclusive, int offset) {
  6881.         if (array == null || startIndexInclusive >= array.length - 1 || endIndexExclusive <= 0) {
  6882.             return;
  6883.         }
  6884.         startIndexInclusive = max0(startIndexInclusive);
  6885.         endIndexExclusive = Math.min(endIndexExclusive, array.length);
  6886.         int n = endIndexExclusive - startIndexInclusive;
  6887.         if (n <= 1) {
  6888.             return;
  6889.         }
  6890.         offset %= n;
  6891.         if (offset < 0) {
  6892.             offset += n;
  6893.         }
  6894.         // For algorithm explanations and proof of O(n) time complexity and O(1) space complexity
  6895.         // see https://beradrian.wordpress.com/2015/04/07/shift-an-array-in-on-in-place/
  6896.         while (n > 1 && offset > 0) {
  6897.             final int nOffset = n - offset;
  6898.             if (offset > nOffset) {
  6899.                 swap(array, startIndexInclusive, startIndexInclusive + n - nOffset,  nOffset);
  6900.                 n = offset;
  6901.                 offset -= nOffset;
  6902.             } else if (offset < nOffset) {
  6903.                 swap(array, startIndexInclusive, startIndexInclusive + nOffset,  offset);
  6904.                 startIndexInclusive += offset;
  6905.                 n = nOffset;
  6906.             } else {
  6907.                 swap(array, startIndexInclusive, startIndexInclusive + nOffset, offset);
  6908.                 break;
  6909.             }
  6910.         }
  6911.     }

  6912.     /**
  6913.      * Shifts the order of the given float array.
  6914.      *
  6915.      * <p>There is no special handling for multi-dimensional arrays. This method
  6916.      * does nothing for {@code null} or empty input arrays.</p>
  6917.      *
  6918.      * @param array  the array to shift, may be {@code null}
  6919.      * @param offset
  6920.      *          The number of positions to rotate the elements.  If the offset is larger than the number of elements to
  6921.      *          rotate, than the effective offset is modulo the number of elements to rotate.
  6922.      * @since 3.5
  6923.      */
  6924.     public static void shift(final float[] array, final int offset) {
  6925.         if (array != null) {
  6926.             shift(array, 0, array.length, offset);
  6927.         }
  6928.     }

  6929.     /**
  6930.      * Shifts the order of a series of elements in the given float array.
  6931.      *
  6932.      * <p>There is no special handling for multi-dimensional arrays. This method
  6933.      * does nothing for {@code null} or empty input arrays.</p>
  6934.      *
  6935.      * @param array
  6936.      *            the array to shift, may be {@code null}
  6937.      * @param startIndexInclusive
  6938.      *            the starting index. Undervalue (&lt;0) is promoted to 0, overvalue (&gt;array.length) results in no
  6939.      *            change.
  6940.      * @param endIndexExclusive
  6941.      *            elements up to endIndex-1 are shifted in the array. Undervalue (&lt; start index) results in no
  6942.      *            change. Overvalue (&gt;array.length) is demoted to array length.
  6943.      * @param offset
  6944.      *          The number of positions to rotate the elements.  If the offset is larger than the number of elements to
  6945.      *          rotate, than the effective offset is modulo the number of elements to rotate.
  6946.      * @since 3.5
  6947.      */
  6948.     public static void shift(final float[] array, int startIndexInclusive, int endIndexExclusive, int offset) {
  6949.         if (array == null || startIndexInclusive >= array.length - 1 || endIndexExclusive <= 0) {
  6950.             return;
  6951.         }
  6952.         startIndexInclusive = max0(startIndexInclusive);
  6953.         endIndexExclusive = Math.min(endIndexExclusive, array.length);
  6954.         int n = endIndexExclusive - startIndexInclusive;
  6955.         if (n <= 1) {
  6956.             return;
  6957.         }
  6958.         offset %= n;
  6959.         if (offset < 0) {
  6960.             offset += n;
  6961.         }
  6962.         // For algorithm explanations and proof of O(n) time complexity and O(1) space complexity
  6963.         // see https://beradrian.wordpress.com/2015/04/07/shift-an-array-in-on-in-place/
  6964.         while (n > 1 && offset > 0) {
  6965.             final int nOffset = n - offset;
  6966.             if (offset > nOffset) {
  6967.                 swap(array, startIndexInclusive, startIndexInclusive + n - nOffset,  nOffset);
  6968.                 n = offset;
  6969.                 offset -= nOffset;
  6970.             } else if (offset < nOffset) {
  6971.                 swap(array, startIndexInclusive, startIndexInclusive + nOffset,  offset);
  6972.                 startIndexInclusive += offset;
  6973.                 n = nOffset;
  6974.             } else {
  6975.                 swap(array, startIndexInclusive, startIndexInclusive + nOffset, offset);
  6976.                 break;
  6977.             }
  6978.         }
  6979.     }

  6980.     /**
  6981.      * Shifts the order of the given int array.
  6982.      *
  6983.      * <p>There is no special handling for multi-dimensional arrays. This method
  6984.      * does nothing for {@code null} or empty input arrays.</p>
  6985.      *
  6986.      * @param array  the array to shift, may be {@code null}
  6987.      * @param offset
  6988.      *          The number of positions to rotate the elements.  If the offset is larger than the number of elements to
  6989.      *          rotate, than the effective offset is modulo the number of elements to rotate.
  6990.      * @since 3.5
  6991.      */
  6992.     public static void shift(final int[] array, final int offset) {
  6993.         if (array != null) {
  6994.             shift(array, 0, array.length, offset);
  6995.         }
  6996.     }

  6997.     /**
  6998.      * Shifts the order of a series of elements in the given int array.
  6999.      *
  7000.      * <p>There is no special handling for multi-dimensional arrays. This method
  7001.      * does nothing for {@code null} or empty input arrays.</p>
  7002.      *
  7003.      * @param array
  7004.      *            the array to shift, may be {@code null}
  7005.      * @param startIndexInclusive
  7006.      *            the starting index. Undervalue (&lt;0) is promoted to 0, overvalue (&gt;array.length) results in no
  7007.      *            change.
  7008.      * @param endIndexExclusive
  7009.      *            elements up to endIndex-1 are shifted in the array. Undervalue (&lt; start index) results in no
  7010.      *            change. Overvalue (&gt;array.length) is demoted to array length.
  7011.      * @param offset
  7012.      *          The number of positions to rotate the elements.  If the offset is larger than the number of elements to
  7013.      *          rotate, than the effective offset is modulo the number of elements to rotate.
  7014.      * @since 3.5
  7015.      */
  7016.     public static void shift(final int[] array, int startIndexInclusive, int endIndexExclusive, int offset) {
  7017.         if (array == null || startIndexInclusive >= array.length - 1 || endIndexExclusive <= 0) {
  7018.             return;
  7019.         }
  7020.         startIndexInclusive = max0(startIndexInclusive);
  7021.         endIndexExclusive = Math.min(endIndexExclusive, array.length);
  7022.         int n = endIndexExclusive - startIndexInclusive;
  7023.         if (n <= 1) {
  7024.             return;
  7025.         }
  7026.         offset %= n;
  7027.         if (offset < 0) {
  7028.             offset += n;
  7029.         }
  7030.         // For algorithm explanations and proof of O(n) time complexity and O(1) space complexity
  7031.         // see https://beradrian.wordpress.com/2015/04/07/shift-an-array-in-on-in-place/
  7032.         while (n > 1 && offset > 0) {
  7033.             final int nOffset = n - offset;
  7034.             if (offset > nOffset) {
  7035.                 swap(array, startIndexInclusive, startIndexInclusive + n - nOffset,  nOffset);
  7036.                 n = offset;
  7037.                 offset -= nOffset;
  7038.             } else if (offset < nOffset) {
  7039.                 swap(array, startIndexInclusive, startIndexInclusive + nOffset,  offset);
  7040.                 startIndexInclusive += offset;
  7041.                 n = nOffset;
  7042.             } else {
  7043.                 swap(array, startIndexInclusive, startIndexInclusive + nOffset, offset);
  7044.                 break;
  7045.             }
  7046.         }
  7047.     }

  7048.     /**
  7049.      * Shifts the order of the given long array.
  7050.      *
  7051.      * <p>There is no special handling for multi-dimensional arrays. This method
  7052.      * does nothing for {@code null} or empty input arrays.</p>
  7053.      *
  7054.      * @param array  the array to shift, may be {@code null}
  7055.      * @param offset
  7056.      *          The number of positions to rotate the elements.  If the offset is larger than the number of elements to
  7057.      *          rotate, than the effective offset is modulo the number of elements to rotate.
  7058.      * @since 3.5
  7059.      */
  7060.     public static void shift(final long[] array, final int offset) {
  7061.         if (array != null) {
  7062.             shift(array, 0, array.length, offset);
  7063.         }
  7064.     }

  7065.     /**
  7066.      * Shifts the order of a series of elements in the given long array.
  7067.      *
  7068.      * <p>There is no special handling for multi-dimensional arrays. This method
  7069.      * does nothing for {@code null} or empty input arrays.</p>
  7070.      *
  7071.      * @param array
  7072.      *            the array to shift, may be {@code null}
  7073.      * @param startIndexInclusive
  7074.      *            the starting index. Undervalue (&lt;0) is promoted to 0, overvalue (&gt;array.length) results in no
  7075.      *            change.
  7076.      * @param endIndexExclusive
  7077.      *            elements up to endIndex-1 are shifted in the array. Undervalue (&lt; start index) results in no
  7078.      *            change. Overvalue (&gt;array.length) is demoted to array length.
  7079.      * @param offset
  7080.      *          The number of positions to rotate the elements.  If the offset is larger than the number of elements to
  7081.      *          rotate, than the effective offset is modulo the number of elements to rotate.
  7082.      * @since 3.5
  7083.      */
  7084.     public static void shift(final long[] array, int startIndexInclusive, int endIndexExclusive, int offset) {
  7085.         if (array == null || startIndexInclusive >= array.length - 1 || endIndexExclusive <= 0) {
  7086.             return;
  7087.         }
  7088.         startIndexInclusive = max0(startIndexInclusive);
  7089.         endIndexExclusive = Math.min(endIndexExclusive, array.length);
  7090.         int n = endIndexExclusive - startIndexInclusive;
  7091.         if (n <= 1) {
  7092.             return;
  7093.         }
  7094.         offset %= n;
  7095.         if (offset < 0) {
  7096.             offset += n;
  7097.         }
  7098.         // For algorithm explanations and proof of O(n) time complexity and O(1) space complexity
  7099.         // see https://beradrian.wordpress.com/2015/04/07/shift-an-array-in-on-in-place/
  7100.         while (n > 1 && offset > 0) {
  7101.             final int nOffset = n - offset;
  7102.             if (offset > nOffset) {
  7103.                 swap(array, startIndexInclusive, startIndexInclusive + n - nOffset,  nOffset);
  7104.                 n = offset;
  7105.                 offset -= nOffset;
  7106.             } else if (offset < nOffset) {
  7107.                 swap(array, startIndexInclusive, startIndexInclusive + nOffset,  offset);
  7108.                 startIndexInclusive += offset;
  7109.                 n = nOffset;
  7110.             } else {
  7111.                 swap(array, startIndexInclusive, startIndexInclusive + nOffset, offset);
  7112.                 break;
  7113.             }
  7114.         }
  7115.     }

  7116.     /**
  7117.      * Shifts the order of the given array.
  7118.      *
  7119.      * <p>There is no special handling for multi-dimensional arrays. This method
  7120.      * does nothing for {@code null} or empty input arrays.</p>
  7121.      *
  7122.      * @param array  the array to shift, may be {@code null}
  7123.      * @param offset
  7124.      *          The number of positions to rotate the elements.  If the offset is larger than the number of elements to
  7125.      *          rotate, than the effective offset is modulo the number of elements to rotate.
  7126.      * @since 3.5
  7127.      */
  7128.     public static void shift(final Object[] array, final int offset) {
  7129.         if (array != null) {
  7130.             shift(array, 0, array.length, offset);
  7131.         }
  7132.     }

  7133.     /**
  7134.      * Shifts the order of a series of elements in the given array.
  7135.      *
  7136.      * <p>There is no special handling for multi-dimensional arrays. This method
  7137.      * does nothing for {@code null} or empty input arrays.</p>
  7138.      *
  7139.      * @param array
  7140.      *            the array to shift, may be {@code null}
  7141.      * @param startIndexInclusive
  7142.      *            the starting index. Undervalue (&lt;0) is promoted to 0, overvalue (&gt;array.length) results in no
  7143.      *            change.
  7144.      * @param endIndexExclusive
  7145.      *            elements up to endIndex-1 are shifted in the array. Undervalue (&lt; start index) results in no
  7146.      *            change. Overvalue (&gt;array.length) is demoted to array length.
  7147.      * @param offset
  7148.      *          The number of positions to rotate the elements.  If the offset is larger than the number of elements to
  7149.      *          rotate, than the effective offset is modulo the number of elements to rotate.
  7150.      * @since 3.5
  7151.      */
  7152.     public static void shift(final Object[] array, int startIndexInclusive, int endIndexExclusive, int offset) {
  7153.         if (array == null || startIndexInclusive >= array.length - 1 || endIndexExclusive <= 0) {
  7154.             return;
  7155.         }
  7156.         startIndexInclusive = max0(startIndexInclusive);
  7157.         endIndexExclusive = Math.min(endIndexExclusive, array.length);
  7158.         int n = endIndexExclusive - startIndexInclusive;
  7159.         if (n <= 1) {
  7160.             return;
  7161.         }
  7162.         offset %= n;
  7163.         if (offset < 0) {
  7164.             offset += n;
  7165.         }
  7166.         // For algorithm explanations and proof of O(n) time complexity and O(1) space complexity
  7167.         // see https://beradrian.wordpress.com/2015/04/07/shift-an-array-in-on-in-place/
  7168.         while (n > 1 && offset > 0) {
  7169.             final int nOffset = n - offset;
  7170.             if (offset > nOffset) {
  7171.                 swap(array, startIndexInclusive, startIndexInclusive + n - nOffset,  nOffset);
  7172.                 n = offset;
  7173.                 offset -= nOffset;
  7174.             } else if (offset < nOffset) {
  7175.                 swap(array, startIndexInclusive, startIndexInclusive + nOffset,  offset);
  7176.                 startIndexInclusive += offset;
  7177.                 n = nOffset;
  7178.             } else {
  7179.                 swap(array, startIndexInclusive, startIndexInclusive + nOffset, offset);
  7180.                 break;
  7181.             }
  7182.         }
  7183.     }

  7184.     /**
  7185.      * Shifts the order of the given short array.
  7186.      *
  7187.      * <p>There is no special handling for multi-dimensional arrays. This method
  7188.      * does nothing for {@code null} or empty input arrays.</p>
  7189.      *
  7190.      * @param array  the array to shift, may be {@code null}
  7191.      * @param offset
  7192.      *          The number of positions to rotate the elements.  If the offset is larger than the number of elements to
  7193.      *          rotate, than the effective offset is modulo the number of elements to rotate.
  7194.      * @since 3.5
  7195.      */
  7196.     public static void shift(final short[] array, final int offset) {
  7197.         if (array != null) {
  7198.             shift(array, 0, array.length, offset);
  7199.         }
  7200.     }

  7201.     /**
  7202.      * Shifts the order of a series of elements in the given short array.
  7203.      *
  7204.      * <p>There is no special handling for multi-dimensional arrays. This method
  7205.      * does nothing for {@code null} or empty input arrays.</p>
  7206.      *
  7207.      * @param array
  7208.      *            the array to shift, may be {@code null}
  7209.      * @param startIndexInclusive
  7210.      *            the starting index. Undervalue (&lt;0) is promoted to 0, overvalue (&gt;array.length) results in no
  7211.      *            change.
  7212.      * @param endIndexExclusive
  7213.      *            elements up to endIndex-1 are shifted in the array. Undervalue (&lt; start index) results in no
  7214.      *            change. Overvalue (&gt;array.length) is demoted to array length.
  7215.      * @param offset
  7216.      *          The number of positions to rotate the elements.  If the offset is larger than the number of elements to
  7217.      *          rotate, than the effective offset is modulo the number of elements to rotate.
  7218.      * @since 3.5
  7219.      */
  7220.     public static void shift(final short[] array, int startIndexInclusive, int endIndexExclusive, int offset) {
  7221.         if (array == null || startIndexInclusive >= array.length - 1 || endIndexExclusive <= 0) {
  7222.             return;
  7223.         }
  7224.         startIndexInclusive = max0(startIndexInclusive);
  7225.         endIndexExclusive = Math.min(endIndexExclusive, array.length);
  7226.         int n = endIndexExclusive - startIndexInclusive;
  7227.         if (n <= 1) {
  7228.             return;
  7229.         }
  7230.         offset %= n;
  7231.         if (offset < 0) {
  7232.             offset += n;
  7233.         }
  7234.         // For algorithm explanations and proof of O(n) time complexity and O(1) space complexity
  7235.         // see https://beradrian.wordpress.com/2015/04/07/shift-an-array-in-on-in-place/
  7236.         while (n > 1 && offset > 0) {
  7237.             final int nOffset = n - offset;
  7238.             if (offset > nOffset) {
  7239.                 swap(array, startIndexInclusive, startIndexInclusive + n - nOffset,  nOffset);
  7240.                 n = offset;
  7241.                 offset -= nOffset;
  7242.             } else if (offset < nOffset) {
  7243.                 swap(array, startIndexInclusive, startIndexInclusive + nOffset,  offset);
  7244.                 startIndexInclusive += offset;
  7245.                 n = nOffset;
  7246.             } else {
  7247.                 swap(array, startIndexInclusive, startIndexInclusive + nOffset, offset);
  7248.                 break;
  7249.             }
  7250.         }
  7251.     }

  7252.     /**
  7253.      * Randomly permutes the elements of the specified array using the Fisher-Yates algorithm.
  7254.      *
  7255.      * @param array   the array to shuffle
  7256.      * @see <a href="https://en.wikipedia.org/wiki/Fisher%E2%80%93Yates_shuffle">Fisher-Yates shuffle algorithm</a>
  7257.      * @since 3.6
  7258.      */
  7259.     public static void shuffle(final boolean[] array) {
  7260.         shuffle(array, random());
  7261.     }

  7262.     /**
  7263.      * Randomly permutes the elements of the specified array using the Fisher-Yates algorithm.
  7264.      *
  7265.      * @param array   the array to shuffle
  7266.      * @param random  the source of randomness used to permute the elements
  7267.      * @see <a href="https://en.wikipedia.org/wiki/Fisher%E2%80%93Yates_shuffle">Fisher-Yates shuffle algorithm</a>
  7268.      * @since 3.6
  7269.      */
  7270.     public static void shuffle(final boolean[] array, final Random random) {
  7271.         for (int i = array.length; i > 1; i--) {
  7272.             swap(array, i - 1, random.nextInt(i), 1);
  7273.         }
  7274.     }

  7275.     /**
  7276.      * Randomly permutes the elements of the specified array using the Fisher-Yates algorithm.
  7277.      *
  7278.      * @param array   the array to shuffle
  7279.      * @see <a href="https://en.wikipedia.org/wiki/Fisher%E2%80%93Yates_shuffle">Fisher-Yates shuffle algorithm</a>
  7280.      * @since 3.6
  7281.      */
  7282.     public static void shuffle(final byte[] array) {
  7283.         shuffle(array, random());
  7284.     }

  7285.     /**
  7286.      * Randomly permutes the elements of the specified array using the Fisher-Yates algorithm.
  7287.      *
  7288.      * @param array   the array to shuffle
  7289.      * @param random  the source of randomness used to permute the elements
  7290.      * @see <a href="https://en.wikipedia.org/wiki/Fisher%E2%80%93Yates_shuffle">Fisher-Yates shuffle algorithm</a>
  7291.      * @since 3.6
  7292.      */
  7293.     public static void shuffle(final byte[] array, final Random random) {
  7294.         for (int i = array.length; i > 1; i--) {
  7295.             swap(array, i - 1, random.nextInt(i), 1);
  7296.         }
  7297.     }

  7298.     /**
  7299.      * Randomly permutes the elements of the specified array using the Fisher-Yates algorithm.
  7300.      *
  7301.      * @param array   the array to shuffle
  7302.      * @see <a href="https://en.wikipedia.org/wiki/Fisher%E2%80%93Yates_shuffle">Fisher-Yates shuffle algorithm</a>
  7303.      * @since 3.6
  7304.      */
  7305.     public static void shuffle(final char[] array) {
  7306.         shuffle(array, random());
  7307.     }

  7308.     /**
  7309.      * Randomly permutes the elements of the specified array using the Fisher-Yates algorithm.
  7310.      *
  7311.      * @param array   the array to shuffle
  7312.      * @param random  the source of randomness used to permute the elements
  7313.      * @see <a href="https://en.wikipedia.org/wiki/Fisher%E2%80%93Yates_shuffle">Fisher-Yates shuffle algorithm</a>
  7314.      * @since 3.6
  7315.      */
  7316.     public static void shuffle(final char[] array, final Random random) {
  7317.         for (int i = array.length; i > 1; i--) {
  7318.             swap(array, i - 1, random.nextInt(i), 1);
  7319.         }
  7320.     }

  7321.     /**
  7322.      * Randomly permutes the elements of the specified array using the Fisher-Yates algorithm.
  7323.      *
  7324.      * @param array   the array to shuffle
  7325.      * @see <a href="https://en.wikipedia.org/wiki/Fisher%E2%80%93Yates_shuffle">Fisher-Yates shuffle algorithm</a>
  7326.      * @since 3.6
  7327.      */
  7328.     public static void shuffle(final double[] array) {
  7329.         shuffle(array, random());
  7330.     }

  7331.     /**
  7332.      * Randomly permutes the elements of the specified array using the Fisher-Yates algorithm.
  7333.      *
  7334.      * @param array   the array to shuffle
  7335.      * @param random  the source of randomness used to permute the elements
  7336.      * @see <a href="https://en.wikipedia.org/wiki/Fisher%E2%80%93Yates_shuffle">Fisher-Yates shuffle algorithm</a>
  7337.      * @since 3.6
  7338.      */
  7339.     public static void shuffle(final double[] array, final Random random) {
  7340.         for (int i = array.length; i > 1; i--) {
  7341.             swap(array, i - 1, random.nextInt(i), 1);
  7342.         }
  7343.     }

  7344.     /**
  7345.      * Randomly permutes the elements of the specified array using the Fisher-Yates algorithm.
  7346.      *
  7347.      * @param array   the array to shuffle
  7348.      * @see <a href="https://en.wikipedia.org/wiki/Fisher%E2%80%93Yates_shuffle">Fisher-Yates shuffle algorithm</a>
  7349.      * @since 3.6
  7350.      */
  7351.     public static void shuffle(final float[] array) {
  7352.         shuffle(array, random());
  7353.     }

  7354.     /**
  7355.      * Randomly permutes the elements of the specified array using the Fisher-Yates algorithm.
  7356.      *
  7357.      * @param array   the array to shuffle
  7358.      * @param random  the source of randomness used to permute the elements
  7359.      * @see <a href="https://en.wikipedia.org/wiki/Fisher%E2%80%93Yates_shuffle">Fisher-Yates shuffle algorithm</a>
  7360.      * @since 3.6
  7361.      */
  7362.     public static void shuffle(final float[] array, final Random random) {
  7363.         for (int i = array.length; i > 1; i--) {
  7364.             swap(array, i - 1, random.nextInt(i), 1);
  7365.         }
  7366.     }

  7367.     /**
  7368.      * Randomly permutes the elements of the specified array using the Fisher-Yates algorithm.
  7369.      *
  7370.      * @param array   the array to shuffle
  7371.      * @see <a href="https://en.wikipedia.org/wiki/Fisher%E2%80%93Yates_shuffle">Fisher-Yates shuffle algorithm</a>
  7372.      * @since 3.6
  7373.      */
  7374.     public static void shuffle(final int[] array) {
  7375.         shuffle(array, random());
  7376.     }

  7377.     /**
  7378.      * Randomly permutes the elements of the specified array using the Fisher-Yates algorithm.
  7379.      *
  7380.      * @param array   the array to shuffle
  7381.      * @param random  the source of randomness used to permute the elements
  7382.      * @see <a href="https://en.wikipedia.org/wiki/Fisher%E2%80%93Yates_shuffle">Fisher-Yates shuffle algorithm</a>
  7383.      * @since 3.6
  7384.      */
  7385.     public static void shuffle(final int[] array, final Random random) {
  7386.         for (int i = array.length; i > 1; i--) {
  7387.             swap(array, i - 1, random.nextInt(i), 1);
  7388.         }
  7389.     }

  7390.     /**
  7391.      * Randomly permutes the elements of the specified array using the Fisher-Yates algorithm.
  7392.      *
  7393.      * @param array   the array to shuffle
  7394.      * @see <a href="https://en.wikipedia.org/wiki/Fisher%E2%80%93Yates_shuffle">Fisher-Yates shuffle algorithm</a>
  7395.      * @since 3.6
  7396.      */
  7397.     public static void shuffle(final long[] array) {
  7398.         shuffle(array, random());
  7399.     }

  7400.     /**
  7401.      * Randomly permutes the elements of the specified array using the Fisher-Yates algorithm.
  7402.      *
  7403.      * @param array   the array to shuffle
  7404.      * @param random  the source of randomness used to permute the elements
  7405.      * @see <a href="https://en.wikipedia.org/wiki/Fisher%E2%80%93Yates_shuffle">Fisher-Yates shuffle algorithm</a>
  7406.      * @since 3.6
  7407.      */
  7408.     public static void shuffle(final long[] array, final Random random) {
  7409.         for (int i = array.length; i > 1; i--) {
  7410.             swap(array, i - 1, random.nextInt(i), 1);
  7411.         }
  7412.     }

  7413.     /**
  7414.      * Randomly permutes the elements of the specified array using the Fisher-Yates algorithm.
  7415.      *
  7416.      * @param array   the array to shuffle
  7417.      * @see <a href="https://en.wikipedia.org/wiki/Fisher%E2%80%93Yates_shuffle">Fisher-Yates shuffle algorithm</a>
  7418.      * @since 3.6
  7419.      */
  7420.     public static void shuffle(final Object[] array) {
  7421.         shuffle(array, random());
  7422.     }

  7423.     /**
  7424.      * Randomly permutes the elements of the specified array using the Fisher-Yates algorithm.
  7425.      *
  7426.      * @param array   the array to shuffle
  7427.      * @param random  the source of randomness used to permute the elements
  7428.      * @see <a href="https://en.wikipedia.org/wiki/Fisher%E2%80%93Yates_shuffle">Fisher-Yates shuffle algorithm</a>
  7429.      * @since 3.6
  7430.      */
  7431.     public static void shuffle(final Object[] array, final Random random) {
  7432.         for (int i = array.length; i > 1; i--) {
  7433.             swap(array, i - 1, random.nextInt(i), 1);
  7434.         }
  7435.     }

  7436.     /**
  7437.      * Randomly permutes the elements of the specified array using the Fisher-Yates algorithm.
  7438.      *
  7439.      * @param array   the array to shuffle
  7440.      * @see <a href="https://en.wikipedia.org/wiki/Fisher%E2%80%93Yates_shuffle">Fisher-Yates shuffle algorithm</a>
  7441.      * @since 3.6
  7442.      */
  7443.     public static void shuffle(final short[] array) {
  7444.         shuffle(array, random());
  7445.     }

  7446.     /**
  7447.      * Randomly permutes the elements of the specified array using the Fisher-Yates algorithm.
  7448.      *
  7449.      * @param array   the array to shuffle
  7450.      * @param random  the source of randomness used to permute the elements
  7451.      * @see <a href="https://en.wikipedia.org/wiki/Fisher%E2%80%93Yates_shuffle">Fisher-Yates shuffle algorithm</a>
  7452.      * @since 3.6
  7453.      */
  7454.     public static void shuffle(final short[] array, final Random random) {
  7455.         for (int i = array.length; i > 1; i--) {
  7456.             swap(array, i - 1, random.nextInt(i), 1);
  7457.         }
  7458.     }

  7459.     /**
  7460.      * Produces a new {@code boolean} array containing the elements
  7461.      * between the start and end indices.
  7462.      * <p>
  7463.      * The start index is inclusive, the end index exclusive.
  7464.      * Null array input produces null output.
  7465.      * </p>
  7466.      *
  7467.      * @param array  the array
  7468.      * @param startIndexInclusive  the starting index. Undervalue (&lt;0)
  7469.      *      is promoted to 0, overvalue (&gt;array.length) results
  7470.      *      in an empty array.
  7471.      * @param endIndexExclusive  elements up to endIndex-1 are present in the
  7472.      *      returned subarray. Undervalue (&lt; startIndex) produces
  7473.      *      empty array, overvalue (&gt;array.length) is demoted to
  7474.      *      array length.
  7475.      * @return a new array containing the elements between
  7476.      *      the start and end indices.
  7477.      * @since 2.1
  7478.      * @see Arrays#copyOfRange(boolean[], int, int)
  7479.      */
  7480.     public static boolean[] subarray(final boolean[] array, int startIndexInclusive, int endIndexExclusive) {
  7481.         if (array == null) {
  7482.             return null;
  7483.         }
  7484.         startIndexInclusive = max0(startIndexInclusive);
  7485.         endIndexExclusive = Math.min(endIndexExclusive, array.length);
  7486.         final int newSize = endIndexExclusive - startIndexInclusive;
  7487.         if (newSize <= 0) {
  7488.             return EMPTY_BOOLEAN_ARRAY;
  7489.         }
  7490.         return arraycopy(array, startIndexInclusive, 0, newSize, boolean[]::new);
  7491.     }

  7492.     /**
  7493.      * Produces a new {@code byte} array containing the elements
  7494.      * between the start and end indices.
  7495.      * <p>
  7496.      * The start index is inclusive, the end index exclusive.
  7497.      * Null array input produces null output.
  7498.      * </p>
  7499.      *
  7500.      * @param array  the array
  7501.      * @param startIndexInclusive  the starting index. Undervalue (&lt;0)
  7502.      *      is promoted to 0, overvalue (&gt;array.length) results
  7503.      *      in an empty array.
  7504.      * @param endIndexExclusive  elements up to endIndex-1 are present in the
  7505.      *      returned subarray. Undervalue (&lt; startIndex) produces
  7506.      *      empty array, overvalue (&gt;array.length) is demoted to
  7507.      *      array length.
  7508.      * @return a new array containing the elements between
  7509.      *      the start and end indices.
  7510.      * @since 2.1
  7511.      * @see Arrays#copyOfRange(byte[], int, int)
  7512.      */
  7513.     public static byte[] subarray(final byte[] array, int startIndexInclusive, int endIndexExclusive) {
  7514.         if (array == null) {
  7515.             return null;
  7516.         }
  7517.         startIndexInclusive = max0(startIndexInclusive);
  7518.         endIndexExclusive = Math.min(endIndexExclusive, array.length);
  7519.         final int newSize = endIndexExclusive - startIndexInclusive;
  7520.         if (newSize <= 0) {
  7521.             return EMPTY_BYTE_ARRAY;
  7522.         }
  7523.         return arraycopy(array, startIndexInclusive, 0, newSize, byte[]::new);
  7524.     }

  7525.     /**
  7526.      * Produces a new {@code char} array containing the elements
  7527.      * between the start and end indices.
  7528.      * <p>
  7529.      * The start index is inclusive, the end index exclusive.
  7530.      * Null array input produces null output.
  7531.      * </p>
  7532.      *
  7533.      * @param array  the array
  7534.      * @param startIndexInclusive  the starting index. Undervalue (&lt;0)
  7535.      *      is promoted to 0, overvalue (&gt;array.length) results
  7536.      *      in an empty array.
  7537.      * @param endIndexExclusive  elements up to endIndex-1 are present in the
  7538.      *      returned subarray. Undervalue (&lt; startIndex) produces
  7539.      *      empty array, overvalue (&gt;array.length) is demoted to
  7540.      *      array length.
  7541.      * @return a new array containing the elements between
  7542.      *      the start and end indices.
  7543.      * @since 2.1
  7544.      * @see Arrays#copyOfRange(char[], int, int)
  7545.      */
  7546.     public static char[] subarray(final char[] array, int startIndexInclusive, int endIndexExclusive) {
  7547.         if (array == null) {
  7548.             return null;
  7549.         }
  7550.         startIndexInclusive = max0(startIndexInclusive);
  7551.         endIndexExclusive = Math.min(endIndexExclusive, array.length);
  7552.         final int newSize = endIndexExclusive - startIndexInclusive;
  7553.         if (newSize <= 0) {
  7554.             return EMPTY_CHAR_ARRAY;
  7555.         }
  7556.         return arraycopy(array, startIndexInclusive, 0, newSize, char[]::new);
  7557.     }

  7558.     /**
  7559.      * Produces a new {@code double} array containing the elements
  7560.      * between the start and end indices.
  7561.      * <p>
  7562.      * The start index is inclusive, the end index exclusive.
  7563.      * Null array input produces null output.
  7564.      * </p>
  7565.      *
  7566.      * @param array  the array
  7567.      * @param startIndexInclusive  the starting index. Undervalue (&lt;0)
  7568.      *      is promoted to 0, overvalue (&gt;array.length) results
  7569.      *      in an empty array.
  7570.      * @param endIndexExclusive  elements up to endIndex-1 are present in the
  7571.      *      returned subarray. Undervalue (&lt; startIndex) produces
  7572.      *      empty array, overvalue (&gt;array.length) is demoted to
  7573.      *      array length.
  7574.      * @return a new array containing the elements between
  7575.      *      the start and end indices.
  7576.      * @since 2.1
  7577.      * @see Arrays#copyOfRange(double[], int, int)
  7578.      */
  7579.     public static double[] subarray(final double[] array, int startIndexInclusive, int endIndexExclusive) {
  7580.         if (array == null) {
  7581.             return null;
  7582.         }
  7583.         startIndexInclusive = max0(startIndexInclusive);
  7584.         endIndexExclusive = Math.min(endIndexExclusive, array.length);
  7585.         final int newSize = endIndexExclusive - startIndexInclusive;
  7586.         if (newSize <= 0) {
  7587.             return EMPTY_DOUBLE_ARRAY;
  7588.         }
  7589.         return arraycopy(array, startIndexInclusive, 0, newSize, double[]::new);
  7590.     }

  7591.     /**
  7592.      * Produces a new {@code float} array containing the elements
  7593.      * between the start and end indices.
  7594.      * <p>
  7595.      * The start index is inclusive, the end index exclusive.
  7596.      * Null array input produces null output.
  7597.      * </p>
  7598.      *
  7599.      * @param array  the array
  7600.      * @param startIndexInclusive  the starting index. Undervalue (&lt;0)
  7601.      *      is promoted to 0, overvalue (&gt;array.length) results
  7602.      *      in an empty array.
  7603.      * @param endIndexExclusive  elements up to endIndex-1 are present in the
  7604.      *      returned subarray. Undervalue (&lt; startIndex) produces
  7605.      *      empty array, overvalue (&gt;array.length) is demoted to
  7606.      *      array length.
  7607.      * @return a new array containing the elements between
  7608.      *      the start and end indices.
  7609.      * @since 2.1
  7610.      * @see Arrays#copyOfRange(float[], int, int)
  7611.      */
  7612.     public static float[] subarray(final float[] array, int startIndexInclusive, int endIndexExclusive) {
  7613.         if (array == null) {
  7614.             return null;
  7615.         }
  7616.         startIndexInclusive = max0(startIndexInclusive);
  7617.         endIndexExclusive = Math.min(endIndexExclusive, array.length);
  7618.         final int newSize = endIndexExclusive - startIndexInclusive;
  7619.         if (newSize <= 0) {
  7620.             return EMPTY_FLOAT_ARRAY;
  7621.         }
  7622.         return arraycopy(array, startIndexInclusive, 0, newSize, float[]::new);
  7623.     }

  7624.     /**
  7625.      * Produces a new {@code int} array containing the elements
  7626.      * between the start and end indices.
  7627.      * <p>
  7628.      * The start index is inclusive, the end index exclusive.
  7629.      * Null array input produces null output.
  7630.      * </p>
  7631.      *
  7632.      * @param array  the array
  7633.      * @param startIndexInclusive  the starting index. Undervalue (&lt;0)
  7634.      *      is promoted to 0, overvalue (&gt;array.length) results
  7635.      *      in an empty array.
  7636.      * @param endIndexExclusive  elements up to endIndex-1 are present in the
  7637.      *      returned subarray. Undervalue (&lt; startIndex) produces
  7638.      *      empty array, overvalue (&gt;array.length) is demoted to
  7639.      *      array length.
  7640.      * @return a new array containing the elements between
  7641.      *      the start and end indices.
  7642.      * @since 2.1
  7643.      * @see Arrays#copyOfRange(int[], int, int)
  7644.      */
  7645.     public static int[] subarray(final int[] array, int startIndexInclusive, int endIndexExclusive) {
  7646.         if (array == null) {
  7647.             return null;
  7648.         }
  7649.         startIndexInclusive = max0(startIndexInclusive);
  7650.         endIndexExclusive = Math.min(endIndexExclusive, array.length);
  7651.         final int newSize = endIndexExclusive - startIndexInclusive;
  7652.         if (newSize <= 0) {
  7653.             return EMPTY_INT_ARRAY;
  7654.         }
  7655.         return arraycopy(array, startIndexInclusive, 0, newSize, int[]::new);
  7656.     }

  7657.     /**
  7658.      * Produces a new {@code long} array containing the elements
  7659.      * between the start and end indices.
  7660.      * <p>
  7661.      * The start index is inclusive, the end index exclusive.
  7662.      * Null array input produces null output.
  7663.      * </p>
  7664.      *
  7665.      * @param array  the array
  7666.      * @param startIndexInclusive  the starting index. Undervalue (&lt;0)
  7667.      *      is promoted to 0, overvalue (&gt;array.length) results
  7668.      *      in an empty array.
  7669.      * @param endIndexExclusive  elements up to endIndex-1 are present in the
  7670.      *      returned subarray. Undervalue (&lt; startIndex) produces
  7671.      *      empty array, overvalue (&gt;array.length) is demoted to
  7672.      *      array length.
  7673.      * @return a new array containing the elements between
  7674.      *      the start and end indices.
  7675.      * @since 2.1
  7676.      * @see Arrays#copyOfRange(long[], int, int)
  7677.      */
  7678.     public static long[] subarray(final long[] array, int startIndexInclusive, int endIndexExclusive) {
  7679.         if (array == null) {
  7680.             return null;
  7681.         }
  7682.         startIndexInclusive = max0(startIndexInclusive);
  7683.         endIndexExclusive = Math.min(endIndexExclusive, array.length);
  7684.         final int newSize = endIndexExclusive - startIndexInclusive;
  7685.         if (newSize <= 0) {
  7686.             return EMPTY_LONG_ARRAY;
  7687.         }
  7688.         return arraycopy(array, startIndexInclusive, 0, newSize, long[]::new);
  7689.     }

  7690.     /**
  7691.      * Produces a new {@code short} array containing the elements
  7692.      * between the start and end indices.
  7693.      * <p>
  7694.      * The start index is inclusive, the end index exclusive.
  7695.      * Null array input produces null output.
  7696.      * </p>
  7697.      *
  7698.      * @param array  the array
  7699.      * @param startIndexInclusive  the starting index. Undervalue (&lt;0)
  7700.      *      is promoted to 0, overvalue (&gt;array.length) results
  7701.      *      in an empty array.
  7702.      * @param endIndexExclusive  elements up to endIndex-1 are present in the
  7703.      *      returned subarray. Undervalue (&lt; startIndex) produces
  7704.      *      empty array, overvalue (&gt;array.length) is demoted to
  7705.      *      array length.
  7706.      * @return a new array containing the elements between
  7707.      *      the start and end indices.
  7708.      * @since 2.1
  7709.      * @see Arrays#copyOfRange(short[], int, int)
  7710.      */
  7711.     public static short[] subarray(final short[] array, int startIndexInclusive, int endIndexExclusive) {
  7712.         if (array == null) {
  7713.             return null;
  7714.         }
  7715.         startIndexInclusive = max0(startIndexInclusive);
  7716.         endIndexExclusive = Math.min(endIndexExclusive, array.length);
  7717.         final int newSize = endIndexExclusive - startIndexInclusive;
  7718.         if (newSize <= 0) {
  7719.             return EMPTY_SHORT_ARRAY;
  7720.         }
  7721.         return arraycopy(array, startIndexInclusive, 0, newSize, short[]::new);
  7722.     }

  7723.     /**
  7724.      * Produces a new array containing the elements between
  7725.      * the start and end indices.
  7726.      * <p>
  7727.      * The start index is inclusive, the end index exclusive.
  7728.      * Null array input produces null output.
  7729.      * </p>
  7730.      * <p>
  7731.      * The component type of the subarray is always the same as
  7732.      * that of the input array. Thus, if the input is an array of type
  7733.      * {@link Date}, the following usage is envisaged:
  7734.      * </p>
  7735.      * <pre>
  7736.      * Date[] someDates = (Date[]) ArrayUtils.subarray(allDates, 2, 5);
  7737.      * </pre>
  7738.      *
  7739.      * @param <T> the component type of the array
  7740.      * @param array  the array
  7741.      * @param startIndexInclusive  the starting index. Undervalue (&lt;0)
  7742.      *      is promoted to 0, overvalue (&gt;array.length) results
  7743.      *      in an empty array.
  7744.      * @param endIndexExclusive  elements up to endIndex-1 are present in the
  7745.      *      returned subarray. Undervalue (&lt; startIndex) produces
  7746.      *      empty array, overvalue (&gt;array.length) is demoted to
  7747.      *      array length.
  7748.      * @return a new array containing the elements between
  7749.      *      the start and end indices.
  7750.      * @since 2.1
  7751.      * @see Arrays#copyOfRange(Object[], int, int)
  7752.      */
  7753.     public static <T> T[] subarray(final T[] array, int startIndexInclusive, int endIndexExclusive) {
  7754.         if (array == null) {
  7755.             return null;
  7756.         }
  7757.         startIndexInclusive = max0(startIndexInclusive);
  7758.         endIndexExclusive = Math.min(endIndexExclusive, array.length);
  7759.         final int newSize = endIndexExclusive - startIndexInclusive;
  7760.         final Class<T> type = getComponentType(array);
  7761.         if (newSize <= 0) {
  7762.             return newInstance(type, 0);
  7763.         }
  7764.         return arraycopy(array, startIndexInclusive, 0, newSize, () -> newInstance(type, newSize));
  7765.     }

  7766.     /**
  7767.      * Swaps two elements in the given boolean array.
  7768.      *
  7769.      * <p>There is no special handling for multi-dimensional arrays. This method
  7770.      * does nothing for a {@code null} or empty input array or for overflow indices.
  7771.      * Negative indices are promoted to 0(zero).</p>
  7772.      *
  7773.      * Examples:
  7774.      * <ul>
  7775.      *     <li>ArrayUtils.swap([1, 2, 3], 0, 2) -&gt; [3, 2, 1]</li>
  7776.      *     <li>ArrayUtils.swap([1, 2, 3], 0, 0) -&gt; [1, 2, 3]</li>
  7777.      *     <li>ArrayUtils.swap([1, 2, 3], 1, 0) -&gt; [2, 1, 3]</li>
  7778.      *     <li>ArrayUtils.swap([1, 2, 3], 0, 5) -&gt; [1, 2, 3]</li>
  7779.      *     <li>ArrayUtils.swap([1, 2, 3], -1, 1) -&gt; [2, 1, 3]</li>
  7780.      * </ul>
  7781.      *
  7782.      * @param array  the array to swap, may be {@code null}
  7783.      * @param offset1 the index of the first element to swap
  7784.      * @param offset2 the index of the second element to swap
  7785.      * @since 3.5
  7786.      */
  7787.     public static void swap(final boolean[] array, final int offset1, final int offset2) {
  7788.         swap(array, offset1, offset2, 1);
  7789.     }

  7790.     /**
  7791.      * Swaps a series of elements in the given boolean array.
  7792.      *
  7793.      * <p>This method does nothing for a {@code null} or empty input array or
  7794.      * for overflow indices. Negative indices are promoted to 0(zero). If any
  7795.      * of the sub-arrays to swap falls outside of the given array, then the
  7796.      * swap is stopped at the end of the array and as many as possible elements
  7797.      * are swapped.</p>
  7798.      *
  7799.      * Examples:
  7800.      * <ul>
  7801.      *     <li>ArrayUtils.swap([true, false, true, false], 0, 2, 1) -&gt; [true, false, true, false]</li>
  7802.      *     <li>ArrayUtils.swap([true, false, true, false], 0, 0, 1) -&gt; [true, false, true, false]</li>
  7803.      *     <li>ArrayUtils.swap([true, false, true, false], 0, 2, 2) -&gt; [true, false, true, false]</li>
  7804.      *     <li>ArrayUtils.swap([true, false, true, false], -3, 2, 2) -&gt; [true, false, true, false]</li>
  7805.      *     <li>ArrayUtils.swap([true, false, true, false], 0, 3, 3) -&gt; [false, false, true, true]</li>
  7806.      * </ul>
  7807.      *
  7808.      * @param array the array to swap, may be {@code null}
  7809.      * @param offset1 the index of the first element in the series to swap
  7810.      * @param offset2 the index of the second element in the series to swap
  7811.      * @param len the number of elements to swap starting with the given indices
  7812.      * @since 3.5
  7813.      */
  7814.     public static void swap(final boolean[] array, int offset1, int offset2, int len) {
  7815.         if (isEmpty(array) || offset1 >= array.length || offset2 >= array.length) {
  7816.             return;
  7817.         }
  7818.         offset1 = max0(offset1);
  7819.         offset2 = max0(offset2);
  7820.         len = Math.min(Math.min(len, array.length - offset1), array.length - offset2);
  7821.         for (int i = 0; i < len; i++, offset1++, offset2++) {
  7822.             final boolean aux = array[offset1];
  7823.             array[offset1] = array[offset2];
  7824.             array[offset2] = aux;
  7825.         }
  7826.     }

  7827.     /**
  7828.      * Swaps two elements in the given byte array.
  7829.      *
  7830.      * <p>There is no special handling for multi-dimensional arrays. This method
  7831.      * does nothing for a {@code null} or empty input array or for overflow indices.
  7832.      * Negative indices are promoted to 0(zero).</p>
  7833.      *
  7834.      * Examples:
  7835.      * <ul>
  7836.      *     <li>ArrayUtils.swap([1, 2, 3], 0, 2) -&gt; [3, 2, 1]</li>
  7837.      *     <li>ArrayUtils.swap([1, 2, 3], 0, 0) -&gt; [1, 2, 3]</li>
  7838.      *     <li>ArrayUtils.swap([1, 2, 3], 1, 0) -&gt; [2, 1, 3]</li>
  7839.      *     <li>ArrayUtils.swap([1, 2, 3], 0, 5) -&gt; [1, 2, 3]</li>
  7840.      *     <li>ArrayUtils.swap([1, 2, 3], -1, 1) -&gt; [2, 1, 3]</li>
  7841.      * </ul>
  7842.      *
  7843.      * @param array  the array to swap, may be {@code null}
  7844.      * @param offset1 the index of the first element to swap
  7845.      * @param offset2 the index of the second element to swap
  7846.      * @since 3.5
  7847.      */
  7848.     public static void swap(final byte[] array, final int offset1, final int offset2) {
  7849.         swap(array, offset1, offset2, 1);
  7850.     }

  7851.     /**
  7852.      * Swaps a series of elements in the given byte array.
  7853.      *
  7854.      * <p>This method does nothing for a {@code null} or empty input array or
  7855.      * for overflow indices. Negative indices are promoted to 0(zero). If any
  7856.      * of the sub-arrays to swap falls outside of the given array, then the
  7857.      * swap is stopped at the end of the array and as many as possible elements
  7858.      * are swapped.</p>
  7859.      *
  7860.      * Examples:
  7861.      * <ul>
  7862.      *     <li>ArrayUtils.swap([1, 2, 3, 4], 0, 2, 1) -&gt; [3, 2, 1, 4]</li>
  7863.      *     <li>ArrayUtils.swap([1, 2, 3, 4], 0, 0, 1) -&gt; [1, 2, 3, 4]</li>
  7864.      *     <li>ArrayUtils.swap([1, 2, 3, 4], 2, 0, 2) -&gt; [3, 4, 1, 2]</li>
  7865.      *     <li>ArrayUtils.swap([1, 2, 3, 4], -3, 2, 2) -&gt; [3, 4, 1, 2]</li>
  7866.      *     <li>ArrayUtils.swap([1, 2, 3, 4], 0, 3, 3) -&gt; [4, 2, 3, 1]</li>
  7867.      * </ul>
  7868.      *
  7869.      * @param array the array to swap, may be {@code null}
  7870.      * @param offset1 the index of the first element in the series to swap
  7871.      * @param offset2 the index of the second element in the series to swap
  7872.      * @param len the number of elements to swap starting with the given indices
  7873.      * @since 3.5
  7874.      */
  7875.     public static void swap(final byte[] array, int offset1, int offset2, int len) {
  7876.         if (isEmpty(array) || offset1 >= array.length || offset2 >= array.length) {
  7877.             return;
  7878.         }
  7879.         offset1 = max0(offset1);
  7880.         offset2 = max0(offset2);
  7881.         len = Math.min(Math.min(len, array.length - offset1), array.length - offset2);
  7882.         for (int i = 0; i < len; i++, offset1++, offset2++) {
  7883.             final byte aux = array[offset1];
  7884.             array[offset1] = array[offset2];
  7885.             array[offset2] = aux;
  7886.         }
  7887.     }

  7888.     /**
  7889.      * Swaps two elements in the given char array.
  7890.      *
  7891.      * <p>There is no special handling for multi-dimensional arrays. This method
  7892.      * does nothing for a {@code null} or empty input array or for overflow indices.
  7893.      * Negative indices are promoted to 0(zero).</p>
  7894.      *
  7895.      * Examples:
  7896.      * <ul>
  7897.      *     <li>ArrayUtils.swap([1, 2, 3], 0, 2) -&gt; [3, 2, 1]</li>
  7898.      *     <li>ArrayUtils.swap([1, 2, 3], 0, 0) -&gt; [1, 2, 3]</li>
  7899.      *     <li>ArrayUtils.swap([1, 2, 3], 1, 0) -&gt; [2, 1, 3]</li>
  7900.      *     <li>ArrayUtils.swap([1, 2, 3], 0, 5) -&gt; [1, 2, 3]</li>
  7901.      *     <li>ArrayUtils.swap([1, 2, 3], -1, 1) -&gt; [2, 1, 3]</li>
  7902.      * </ul>
  7903.      *
  7904.      * @param array  the array to swap, may be {@code null}
  7905.      * @param offset1 the index of the first element to swap
  7906.      * @param offset2 the index of the second element to swap
  7907.      * @since 3.5
  7908.      */
  7909.     public static void swap(final char[] array, final int offset1, final int offset2) {
  7910.         swap(array, offset1, offset2, 1);
  7911.     }

  7912.     /**
  7913.      * Swaps a series of elements in the given char array.
  7914.      *
  7915.      * <p>This method does nothing for a {@code null} or empty input array or
  7916.      * for overflow indices. Negative indices are promoted to 0(zero). If any
  7917.      * of the sub-arrays to swap falls outside of the given array, then the
  7918.      * swap is stopped at the end of the array and as many as possible elements
  7919.      * are swapped.</p>
  7920.      *
  7921.      * Examples:
  7922.      * <ul>
  7923.      *     <li>ArrayUtils.swap([1, 2, 3, 4], 0, 2, 1) -&gt; [3, 2, 1, 4]</li>
  7924.      *     <li>ArrayUtils.swap([1, 2, 3, 4], 0, 0, 1) -&gt; [1, 2, 3, 4]</li>
  7925.      *     <li>ArrayUtils.swap([1, 2, 3, 4], 2, 0, 2) -&gt; [3, 4, 1, 2]</li>
  7926.      *     <li>ArrayUtils.swap([1, 2, 3, 4], -3, 2, 2) -&gt; [3, 4, 1, 2]</li>
  7927.      *     <li>ArrayUtils.swap([1, 2, 3, 4], 0, 3, 3) -&gt; [4, 2, 3, 1]</li>
  7928.      * </ul>
  7929.      *
  7930.      * @param array the array to swap, may be {@code null}
  7931.      * @param offset1 the index of the first element in the series to swap
  7932.      * @param offset2 the index of the second element in the series to swap
  7933.      * @param len the number of elements to swap starting with the given indices
  7934.      * @since 3.5
  7935.      */
  7936.     public static void swap(final char[] array, int offset1, int offset2, int len) {
  7937.         if (isEmpty(array) || offset1 >= array.length || offset2 >= array.length) {
  7938.             return;
  7939.         }
  7940.         offset1 = max0(offset1);
  7941.         offset2 = max0(offset2);
  7942.         len = Math.min(Math.min(len, array.length - offset1), array.length - offset2);
  7943.         for (int i = 0; i < len; i++, offset1++, offset2++) {
  7944.             final char aux = array[offset1];
  7945.             array[offset1] = array[offset2];
  7946.             array[offset2] = aux;
  7947.         }
  7948.     }

  7949.     /**
  7950.      * Swaps two elements in the given double array.
  7951.      *
  7952.      * <p>There is no special handling for multi-dimensional arrays. This method
  7953.      * does nothing for a {@code null} or empty input array or for overflow indices.
  7954.      * Negative indices are promoted to 0(zero).</p>
  7955.      *
  7956.      * Examples:
  7957.      * <ul>
  7958.      *     <li>ArrayUtils.swap([1, 2, 3], 0, 2) -&gt; [3, 2, 1]</li>
  7959.      *     <li>ArrayUtils.swap([1, 2, 3], 0, 0) -&gt; [1, 2, 3]</li>
  7960.      *     <li>ArrayUtils.swap([1, 2, 3], 1, 0) -&gt; [2, 1, 3]</li>
  7961.      *     <li>ArrayUtils.swap([1, 2, 3], 0, 5) -&gt; [1, 2, 3]</li>
  7962.      *     <li>ArrayUtils.swap([1, 2, 3], -1, 1) -&gt; [2, 1, 3]</li>
  7963.      * </ul>
  7964.      *
  7965.      * @param array  the array to swap, may be {@code null}
  7966.      * @param offset1 the index of the first element to swap
  7967.      * @param offset2 the index of the second element to swap
  7968.      * @since 3.5
  7969.      */
  7970.     public static void swap(final double[] array, final int offset1, final int offset2) {
  7971.         swap(array, offset1, offset2, 1);
  7972.     }

  7973.     /**
  7974.      * Swaps a series of elements in the given double array.
  7975.      *
  7976.      * <p>This method does nothing for a {@code null} or empty input array or
  7977.      * for overflow indices. Negative indices are promoted to 0(zero). If any
  7978.      * of the sub-arrays to swap falls outside of the given array, then the
  7979.      * swap is stopped at the end of the array and as many as possible elements
  7980.      * are swapped.</p>
  7981.      *
  7982.      * Examples:
  7983.      * <ul>
  7984.      *     <li>ArrayUtils.swap([1, 2, 3, 4], 0, 2, 1) -&gt; [3, 2, 1, 4]</li>
  7985.      *     <li>ArrayUtils.swap([1, 2, 3, 4], 0, 0, 1) -&gt; [1, 2, 3, 4]</li>
  7986.      *     <li>ArrayUtils.swap([1, 2, 3, 4], 2, 0, 2) -&gt; [3, 4, 1, 2]</li>
  7987.      *     <li>ArrayUtils.swap([1, 2, 3, 4], -3, 2, 2) -&gt; [3, 4, 1, 2]</li>
  7988.      *     <li>ArrayUtils.swap([1, 2, 3, 4], 0, 3, 3) -&gt; [4, 2, 3, 1]</li>
  7989.      * </ul>
  7990.      *
  7991.      * @param array the array to swap, may be {@code null}
  7992.      * @param offset1 the index of the first element in the series to swap
  7993.      * @param offset2 the index of the second element in the series to swap
  7994.      * @param len the number of elements to swap starting with the given indices
  7995.      * @since 3.5
  7996.      */
  7997.     public static void swap(final double[] array,  int offset1, int offset2, int len) {
  7998.         if (isEmpty(array) || offset1 >= array.length || offset2 >= array.length) {
  7999.             return;
  8000.         }
  8001.         offset1 = max0(offset1);
  8002.         offset2 = max0(offset2);
  8003.         len = Math.min(Math.min(len, array.length - offset1), array.length - offset2);
  8004.         for (int i = 0; i < len; i++, offset1++, offset2++) {
  8005.             final double aux = array[offset1];
  8006.             array[offset1] = array[offset2];
  8007.             array[offset2] = aux;
  8008.         }
  8009.     }

  8010.     /**
  8011.      * Swaps two elements in the given float array.
  8012.      *
  8013.      * <p>There is no special handling for multi-dimensional arrays. This method
  8014.      * does nothing for a {@code null} or empty input array or for overflow indices.
  8015.      * Negative indices are promoted to 0(zero).</p>
  8016.      *
  8017.      * Examples:
  8018.      * <ul>
  8019.      *     <li>ArrayUtils.swap([1, 2, 3], 0, 2) -&gt; [3, 2, 1]</li>
  8020.      *     <li>ArrayUtils.swap([1, 2, 3], 0, 0) -&gt; [1, 2, 3]</li>
  8021.      *     <li>ArrayUtils.swap([1, 2, 3], 1, 0) -&gt; [2, 1, 3]</li>
  8022.      *     <li>ArrayUtils.swap([1, 2, 3], 0, 5) -&gt; [1, 2, 3]</li>
  8023.      *     <li>ArrayUtils.swap([1, 2, 3], -1, 1) -&gt; [2, 1, 3]</li>
  8024.      * </ul>
  8025.      *
  8026.      * @param array  the array to swap, may be {@code null}
  8027.      * @param offset1 the index of the first element to swap
  8028.      * @param offset2 the index of the second element to swap
  8029.      * @since 3.5
  8030.      */
  8031.     public static void swap(final float[] array, final int offset1, final int offset2) {
  8032.         swap(array, offset1, offset2, 1);
  8033.     }

  8034.     /**
  8035.      * Swaps a series of elements in the given float array.
  8036.      *
  8037.      * <p>This method does nothing for a {@code null} or empty input array or
  8038.      * for overflow indices. Negative indices are promoted to 0(zero). If any
  8039.      * of the sub-arrays to swap falls outside of the given array, then the
  8040.      * swap is stopped at the end of the array and as many as possible elements
  8041.      * are swapped.</p>
  8042.      *
  8043.      * Examples:
  8044.      * <ul>
  8045.      *     <li>ArrayUtils.swap([1, 2, 3, 4], 0, 2, 1) -&gt; [3, 2, 1, 4]</li>
  8046.      *     <li>ArrayUtils.swap([1, 2, 3, 4], 0, 0, 1) -&gt; [1, 2, 3, 4]</li>
  8047.      *     <li>ArrayUtils.swap([1, 2, 3, 4], 2, 0, 2) -&gt; [3, 4, 1, 2]</li>
  8048.      *     <li>ArrayUtils.swap([1, 2, 3, 4], -3, 2, 2) -&gt; [3, 4, 1, 2]</li>
  8049.      *     <li>ArrayUtils.swap([1, 2, 3, 4], 0, 3, 3) -&gt; [4, 2, 3, 1]</li>
  8050.      * </ul>
  8051.      *
  8052.      * @param array the array to swap, may be {@code null}
  8053.      * @param offset1 the index of the first element in the series to swap
  8054.      * @param offset2 the index of the second element in the series to swap
  8055.      * @param len the number of elements to swap starting with the given indices
  8056.      * @since 3.5
  8057.      */
  8058.     public static void swap(final float[] array, int offset1, int offset2, int len) {
  8059.         if (isEmpty(array) || offset1 >= array.length || offset2 >= array.length) {
  8060.             return;
  8061.         }
  8062.         offset1 = max0(offset1);
  8063.         offset2 = max0(offset2);
  8064.         len = Math.min(Math.min(len, array.length - offset1), array.length - offset2);
  8065.         for (int i = 0; i < len; i++, offset1++, offset2++) {
  8066.             final float aux = array[offset1];
  8067.             array[offset1] = array[offset2];
  8068.             array[offset2] = aux;
  8069.         }

  8070.     }

  8071.     /**
  8072.      * Swaps two elements in the given int array.
  8073.      *
  8074.      * <p>There is no special handling for multi-dimensional arrays. This method
  8075.      * does nothing for a {@code null} or empty input array or for overflow indices.
  8076.      * Negative indices are promoted to 0(zero).</p>
  8077.      *
  8078.      * Examples:
  8079.      * <ul>
  8080.      *     <li>ArrayUtils.swap([1, 2, 3], 0, 2) -&gt; [3, 2, 1]</li>
  8081.      *     <li>ArrayUtils.swap([1, 2, 3], 0, 0) -&gt; [1, 2, 3]</li>
  8082.      *     <li>ArrayUtils.swap([1, 2, 3], 1, 0) -&gt; [2, 1, 3]</li>
  8083.      *     <li>ArrayUtils.swap([1, 2, 3], 0, 5) -&gt; [1, 2, 3]</li>
  8084.      *     <li>ArrayUtils.swap([1, 2, 3], -1, 1) -&gt; [2, 1, 3]</li>
  8085.      * </ul>
  8086.      *
  8087.      * @param array  the array to swap, may be {@code null}
  8088.      * @param offset1 the index of the first element to swap
  8089.      * @param offset2 the index of the second element to swap
  8090.      * @since 3.5
  8091.      */
  8092.     public static void swap(final int[] array, final int offset1, final int offset2) {
  8093.         swap(array, offset1, offset2, 1);
  8094.     }

  8095.     /**
  8096.      * Swaps a series of elements in the given int array.
  8097.      *
  8098.      * <p>This method does nothing for a {@code null} or empty input array or
  8099.      * for overflow indices. Negative indices are promoted to 0(zero). If any
  8100.      * of the sub-arrays to swap falls outside of the given array, then the
  8101.      * swap is stopped at the end of the array and as many as possible elements
  8102.      * are swapped.</p>
  8103.      *
  8104.      * Examples:
  8105.      * <ul>
  8106.      *     <li>ArrayUtils.swap([1, 2, 3, 4], 0, 2, 1) -&gt; [3, 2, 1, 4]</li>
  8107.      *     <li>ArrayUtils.swap([1, 2, 3, 4], 0, 0, 1) -&gt; [1, 2, 3, 4]</li>
  8108.      *     <li>ArrayUtils.swap([1, 2, 3, 4], 2, 0, 2) -&gt; [3, 4, 1, 2]</li>
  8109.      *     <li>ArrayUtils.swap([1, 2, 3, 4], -3, 2, 2) -&gt; [3, 4, 1, 2]</li>
  8110.      *     <li>ArrayUtils.swap([1, 2, 3, 4], 0, 3, 3) -&gt; [4, 2, 3, 1]</li>
  8111.      * </ul>
  8112.      *
  8113.      * @param array the array to swap, may be {@code null}
  8114.      * @param offset1 the index of the first element in the series to swap
  8115.      * @param offset2 the index of the second element in the series to swap
  8116.      * @param len the number of elements to swap starting with the given indices
  8117.      * @since 3.5
  8118.      */
  8119.     public static void swap(final int[] array,  int offset1, int offset2, int len) {
  8120.         if (isEmpty(array) || offset1 >= array.length || offset2 >= array.length) {
  8121.             return;
  8122.         }
  8123.         offset1 = max0(offset1);
  8124.         offset2 = max0(offset2);
  8125.         len = Math.min(Math.min(len, array.length - offset1), array.length - offset2);
  8126.         for (int i = 0; i < len; i++, offset1++, offset2++) {
  8127.             final int aux = array[offset1];
  8128.             array[offset1] = array[offset2];
  8129.             array[offset2] = aux;
  8130.         }
  8131.     }

  8132.     /**
  8133.      * Swaps two elements in the given long array.
  8134.      *
  8135.      * <p>There is no special handling for multi-dimensional arrays. This method
  8136.      * does nothing for a {@code null} or empty input array or for overflow indices.
  8137.      * Negative indices are promoted to 0(zero).</p>
  8138.      *
  8139.      * Examples:
  8140.      * <ul>
  8141.      *     <li>ArrayUtils.swap([true, false, true], 0, 2) -&gt; [true, false, true]</li>
  8142.      *     <li>ArrayUtils.swap([true, false, true], 0, 0) -&gt; [true, false, true]</li>
  8143.      *     <li>ArrayUtils.swap([true, false, true], 1, 0) -&gt; [false, true, true]</li>
  8144.      *     <li>ArrayUtils.swap([true, false, true], 0, 5) -&gt; [true, false, true]</li>
  8145.      *     <li>ArrayUtils.swap([true, false, true], -1, 1) -&gt; [false, true, true]</li>
  8146.      * </ul>
  8147.      *
  8148.      * @param array  the array to swap, may be {@code null}
  8149.      * @param offset1 the index of the first element to swap
  8150.      * @param offset2 the index of the second element to swap
  8151.      * @since 3.5
  8152.      */
  8153.     public static void swap(final long[] array, final int offset1, final int offset2) {
  8154.         swap(array, offset1, offset2, 1);
  8155.     }

  8156.     /**
  8157.      * Swaps a series of elements in the given long array.
  8158.      *
  8159.      * <p>This method does nothing for a {@code null} or empty input array or
  8160.      * for overflow indices. Negative indices are promoted to 0(zero). If any
  8161.      * of the sub-arrays to swap falls outside of the given array, then the
  8162.      * swap is stopped at the end of the array and as many as possible elements
  8163.      * are swapped.</p>
  8164.      *
  8165.      * Examples:
  8166.      * <ul>
  8167.      *     <li>ArrayUtils.swap([1, 2, 3, 4], 0, 2, 1) -&gt; [3, 2, 1, 4]</li>
  8168.      *     <li>ArrayUtils.swap([1, 2, 3, 4], 0, 0, 1) -&gt; [1, 2, 3, 4]</li>
  8169.      *     <li>ArrayUtils.swap([1, 2, 3, 4], 2, 0, 2) -&gt; [3, 4, 1, 2]</li>
  8170.      *     <li>ArrayUtils.swap([1, 2, 3, 4], -3, 2, 2) -&gt; [3, 4, 1, 2]</li>
  8171.      *     <li>ArrayUtils.swap([1, 2, 3, 4], 0, 3, 3) -&gt; [4, 2, 3, 1]</li>
  8172.      * </ul>
  8173.      *
  8174.      * @param array the array to swap, may be {@code null}
  8175.      * @param offset1 the index of the first element in the series to swap
  8176.      * @param offset2 the index of the second element in the series to swap
  8177.      * @param len the number of elements to swap starting with the given indices
  8178.      * @since 3.5
  8179.      */
  8180.     public static void swap(final long[] array,  int offset1, int offset2, int len) {
  8181.         if (isEmpty(array) || offset1 >= array.length || offset2 >= array.length) {
  8182.             return;
  8183.         }
  8184.         offset1 = max0(offset1);
  8185.         offset2 = max0(offset2);
  8186.         len = Math.min(Math.min(len, array.length - offset1), array.length - offset2);
  8187.         for (int i = 0; i < len; i++, offset1++, offset2++) {
  8188.             final long aux = array[offset1];
  8189.             array[offset1] = array[offset2];
  8190.             array[offset2] = aux;
  8191.         }
  8192.     }

  8193.     /**
  8194.      * Swaps two elements in the given array.
  8195.      *
  8196.      * <p>There is no special handling for multi-dimensional arrays. This method
  8197.      * does nothing for a {@code null} or empty input array or for overflow indices.
  8198.      * Negative indices are promoted to 0(zero).</p>
  8199.      *
  8200.      * Examples:
  8201.      * <ul>
  8202.      *     <li>ArrayUtils.swap(["1", "2", "3"], 0, 2) -&gt; ["3", "2", "1"]</li>
  8203.      *     <li>ArrayUtils.swap(["1", "2", "3"], 0, 0) -&gt; ["1", "2", "3"]</li>
  8204.      *     <li>ArrayUtils.swap(["1", "2", "3"], 1, 0) -&gt; ["2", "1", "3"]</li>
  8205.      *     <li>ArrayUtils.swap(["1", "2", "3"], 0, 5) -&gt; ["1", "2", "3"]</li>
  8206.      *     <li>ArrayUtils.swap(["1", "2", "3"], -1, 1) -&gt; ["2", "1", "3"]</li>
  8207.      * </ul>
  8208.      *
  8209.      * @param array the array to swap, may be {@code null}
  8210.      * @param offset1 the index of the first element to swap
  8211.      * @param offset2 the index of the second element to swap
  8212.      * @since 3.5
  8213.      */
  8214.     public static void swap(final Object[] array, final int offset1, final int offset2) {
  8215.         swap(array, offset1, offset2, 1);
  8216.     }

  8217.     /**
  8218.      * Swaps a series of elements in the given array.
  8219.      *
  8220.      * <p>This method does nothing for a {@code null} or empty input array or
  8221.      * for overflow indices. Negative indices are promoted to 0(zero). If any
  8222.      * of the sub-arrays to swap falls outside of the given array, then the
  8223.      * swap is stopped at the end of the array and as many as possible elements
  8224.      * are swapped.</p>
  8225.      *
  8226.      * Examples:
  8227.      * <ul>
  8228.      *     <li>ArrayUtils.swap(["1", "2", "3", "4"], 0, 2, 1) -&gt; ["3", "2", "1", "4"]</li>
  8229.      *     <li>ArrayUtils.swap(["1", "2", "3", "4"], 0, 0, 1) -&gt; ["1", "2", "3", "4"]</li>
  8230.      *     <li>ArrayUtils.swap(["1", "2", "3", "4"], 2, 0, 2) -&gt; ["3", "4", "1", "2"]</li>
  8231.      *     <li>ArrayUtils.swap(["1", "2", "3", "4"], -3, 2, 2) -&gt; ["3", "4", "1", "2"]</li>
  8232.      *     <li>ArrayUtils.swap(["1", "2", "3", "4"], 0, 3, 3) -&gt; ["4", "2", "3", "1"]</li>
  8233.      * </ul>
  8234.      *
  8235.      * @param array the array to swap, may be {@code null}
  8236.      * @param offset1 the index of the first element in the series to swap
  8237.      * @param offset2 the index of the second element in the series to swap
  8238.      * @param len the number of elements to swap starting with the given indices
  8239.      * @since 3.5
  8240.      */
  8241.     public static void swap(final Object[] array,  int offset1, int offset2, int len) {
  8242.         if (isEmpty(array) || offset1 >= array.length || offset2 >= array.length) {
  8243.             return;
  8244.         }
  8245.         offset1 = max0(offset1);
  8246.         offset2 = max0(offset2);
  8247.         len = Math.min(Math.min(len, array.length - offset1), array.length - offset2);
  8248.         for (int i = 0; i < len; i++, offset1++, offset2++) {
  8249.             final Object aux = array[offset1];
  8250.             array[offset1] = array[offset2];
  8251.             array[offset2] = aux;
  8252.         }
  8253.     }

  8254.     /**
  8255.      * Swaps two elements in the given short array.
  8256.      *
  8257.      * <p>There is no special handling for multi-dimensional arrays. This method
  8258.      * does nothing for a {@code null} or empty input array or for overflow indices.
  8259.      * Negative indices are promoted to 0(zero).</p>
  8260.      *
  8261.      * Examples:
  8262.      * <ul>
  8263.      *     <li>ArrayUtils.swap([1, 2, 3], 0, 2) -&gt; [3, 2, 1]</li>
  8264.      *     <li>ArrayUtils.swap([1, 2, 3], 0, 0) -&gt; [1, 2, 3]</li>
  8265.      *     <li>ArrayUtils.swap([1, 2, 3], 1, 0) -&gt; [2, 1, 3]</li>
  8266.      *     <li>ArrayUtils.swap([1, 2, 3], 0, 5) -&gt; [1, 2, 3]</li>
  8267.      *     <li>ArrayUtils.swap([1, 2, 3], -1, 1) -&gt; [2, 1, 3]</li>
  8268.      * </ul>
  8269.      *
  8270.      * @param array  the array to swap, may be {@code null}
  8271.      * @param offset1 the index of the first element to swap
  8272.      * @param offset2 the index of the second element to swap
  8273.      * @since 3.5
  8274.      */
  8275.     public static void swap(final short[] array, final int offset1, final int offset2) {
  8276.         swap(array, offset1, offset2, 1);
  8277.     }

  8278.     /**
  8279.      * Swaps a series of elements in the given short array.
  8280.      *
  8281.      * <p>This method does nothing for a {@code null} or empty input array or
  8282.      * for overflow indices. Negative indices are promoted to 0(zero). If any
  8283.      * of the sub-arrays to swap falls outside of the given array, then the
  8284.      * swap is stopped at the end of the array and as many as possible elements
  8285.      * are swapped.</p>
  8286.      *
  8287.      * Examples:
  8288.      * <ul>
  8289.      *     <li>ArrayUtils.swap([1, 2, 3, 4], 0, 2, 1) -&gt; [3, 2, 1, 4]</li>
  8290.      *     <li>ArrayUtils.swap([1, 2, 3, 4], 0, 0, 1) -&gt; [1, 2, 3, 4]</li>
  8291.      *     <li>ArrayUtils.swap([1, 2, 3, 4], 2, 0, 2) -&gt; [3, 4, 1, 2]</li>
  8292.      *     <li>ArrayUtils.swap([1, 2, 3, 4], -3, 2, 2) -&gt; [3, 4, 1, 2]</li>
  8293.      *     <li>ArrayUtils.swap([1, 2, 3, 4], 0, 3, 3) -&gt; [4, 2, 3, 1]</li>
  8294.      * </ul>
  8295.      *
  8296.      * @param array the array to swap, may be {@code null}
  8297.      * @param offset1 the index of the first element in the series to swap
  8298.      * @param offset2 the index of the second element in the series to swap
  8299.      * @param len the number of elements to swap starting with the given indices
  8300.      * @since 3.5
  8301.      */
  8302.     public static void swap(final short[] array, int offset1, int offset2, int len) {
  8303.         if (isEmpty(array) || offset1 >= array.length || offset2 >= array.length) {
  8304.             return;
  8305.         }
  8306.         offset1 = max0(offset1);
  8307.         offset2 = max0(offset2);
  8308.         if (offset1 == offset2) {
  8309.             return;
  8310.         }
  8311.         len = Math.min(Math.min(len, array.length - offset1), array.length - offset2);
  8312.         for (int i = 0; i < len; i++, offset1++, offset2++) {
  8313.             final short aux = array[offset1];
  8314.             array[offset1] = array[offset2];
  8315.             array[offset2] = aux;
  8316.         }
  8317.     }

  8318.     /**
  8319.      * Create a type-safe generic array.
  8320.      * <p>
  8321.      * The Java language does not allow an array to be created from a generic type:
  8322.      * </p>
  8323.      * <pre>
  8324.     public static &lt;T&gt; T[] createAnArray(int size) {
  8325.         return new T[size]; // compiler error here
  8326.     }
  8327.     public static &lt;T&gt; T[] createAnArray(int size) {
  8328.         return (T[]) new Object[size]; // ClassCastException at runtime
  8329.     }
  8330.      * </pre>
  8331.      * <p>
  8332.      * Therefore new arrays of generic types can be created with this method.
  8333.      * For example, an array of Strings can be created:
  8334.      * </p>
  8335.      * <pre>{@code
  8336.      * String[] array = ArrayUtils.toArray("1", "2");
  8337.      * String[] emptyArray = ArrayUtils.<String>toArray();
  8338.      * }</pre>
  8339.      * <p>
  8340.      * The method is typically used in scenarios, where the caller itself uses generic types
  8341.      * that have to be combined into an array.
  8342.      * </p>
  8343.      * <p>
  8344.      * Note, this method makes only sense to provide arguments of the same type so that the
  8345.      * compiler can deduce the type of the array itself. While it is possible to select the
  8346.      * type explicitly like in
  8347.      * {@code Number[] array = ArrayUtils.<Number>toArray(Integer.valueOf(42), Double.valueOf(Math.PI))},
  8348.      * there is no real advantage when compared to
  8349.      * {@code new Number[] {Integer.valueOf(42), Double.valueOf(Math.PI)}}.
  8350.      * </p>
  8351.      *
  8352.      * @param  <T>   the array's element type
  8353.      * @param  items  the varargs array items, null allowed
  8354.      * @return the array, not null unless a null array is passed in
  8355.      * @since 3.0
  8356.      */
  8357.     public static <T> T[] toArray(@SuppressWarnings("unchecked") final T... items) {
  8358.         return items;
  8359.     }

  8360.     /**
  8361.      * Converts the given array into a {@link java.util.Map}. Each element of the array
  8362.      * must be either a {@link java.util.Map.Entry} or an Array, containing at least two
  8363.      * elements, where the first element is used as key and the second as
  8364.      * value.
  8365.      * <p>
  8366.      * This method can be used to initialize:
  8367.      * </p>
  8368.      * <pre>
  8369.      * // Create a Map mapping colors.
  8370.      * Map colorMap = ArrayUtils.toMap(new String[][] {
  8371.      *     {"RED", "#FF0000"},
  8372.      *     {"GREEN", "#00FF00"},
  8373.      *     {"BLUE", "#0000FF"}});
  8374.      * </pre>
  8375.      * <p>
  8376.      * This method returns {@code null} for a {@code null} input array.
  8377.      * </p>
  8378.      *
  8379.      * @param array  an array whose elements are either a {@link java.util.Map.Entry} or
  8380.      *  an Array containing at least two elements, may be {@code null}
  8381.      * @return a {@link Map} that was created from the array
  8382.      * @throws IllegalArgumentException  if one element of this Array is
  8383.      *  itself an Array containing less than two elements
  8384.      * @throws IllegalArgumentException  if the array contains elements other
  8385.      *  than {@link java.util.Map.Entry} and an Array
  8386.      */
  8387.     public static Map<Object, Object> toMap(final Object[] array) {
  8388.         if (array == null) {
  8389.             return null;
  8390.         }
  8391.         final Map<Object, Object> map = new HashMap<>((int) (array.length * 1.5));
  8392.         for (int i = 0; i < array.length; i++) {
  8393.             final Object object = array[i];
  8394.             if (object instanceof Map.Entry<?, ?>) {
  8395.                 final Map.Entry<?, ?> entry = (Map.Entry<?, ?>) object;
  8396.                 map.put(entry.getKey(), entry.getValue());
  8397.             } else if (object instanceof Object[]) {
  8398.                 final Object[] entry = (Object[]) object;
  8399.                 if (entry.length < 2) {
  8400.                     throw new IllegalArgumentException("Array element " + i + ", '"
  8401.                         + object
  8402.                         + "', has a length less than 2");
  8403.                 }
  8404.                 map.put(entry[0], entry[1]);
  8405.             } else {
  8406.                 throw new IllegalArgumentException("Array element " + i + ", '"
  8407.                         + object
  8408.                         + "', is neither of type Map.Entry nor an Array");
  8409.             }
  8410.         }
  8411.         return map;
  8412.     }

  8413.     /**
  8414.      * Converts an array of primitive booleans to objects.
  8415.      *
  8416.      * <p>This method returns {@code null} for a {@code null} input array.</p>
  8417.      *
  8418.      * @param array  a {@code boolean} array
  8419.      * @return a {@link Boolean} array, {@code null} if null array input
  8420.      */
  8421.     public static Boolean[] toObject(final boolean[] array) {
  8422.         if (array == null) {
  8423.             return null;
  8424.         }
  8425.         if (array.length == 0) {
  8426.             return EMPTY_BOOLEAN_OBJECT_ARRAY;
  8427.         }
  8428.         final Boolean[] result = new Boolean[array.length];
  8429.         return setAll(result, i -> array[i] ? Boolean.TRUE : Boolean.FALSE);
  8430.     }

  8431.     /**
  8432.      * Converts an array of primitive bytes to objects.
  8433.      *
  8434.      * <p>This method returns {@code null} for a {@code null} input array.</p>
  8435.      *
  8436.      * @param array  a {@code byte} array
  8437.      * @return a {@link Byte} array, {@code null} if null array input
  8438.      */
  8439.     public static Byte[] toObject(final byte[] array) {
  8440.         if (array == null) {
  8441.             return null;
  8442.         }
  8443.         if (array.length == 0) {
  8444.             return EMPTY_BYTE_OBJECT_ARRAY;
  8445.         }
  8446.         return setAll(new Byte[array.length], i -> Byte.valueOf(array[i]));
  8447.     }

  8448.     /**
  8449.      * Converts an array of primitive chars to objects.
  8450.      *
  8451.      * <p>This method returns {@code null} for a {@code null} input array.</p>
  8452.      *
  8453.      * @param array a {@code char} array
  8454.      * @return a {@link Character} array, {@code null} if null array input
  8455.      */
  8456.     public static Character[] toObject(final char[] array) {
  8457.         if (array == null) {
  8458.             return null;
  8459.         }
  8460.         if (array.length == 0) {
  8461.             return EMPTY_CHARACTER_OBJECT_ARRAY;
  8462.         }
  8463.         return setAll(new Character[array.length], i -> Character.valueOf(array[i]));
  8464.      }

  8465.     /**
  8466.      * Converts an array of primitive doubles to objects.
  8467.      *
  8468.      * <p>This method returns {@code null} for a {@code null} input array.</p>
  8469.      *
  8470.      * @param array  a {@code double} array
  8471.      * @return a {@link Double} array, {@code null} if null array input
  8472.      */
  8473.     public static Double[] toObject(final double[] array) {
  8474.         if (array == null) {
  8475.             return null;
  8476.         }
  8477.         if (array.length == 0) {
  8478.             return EMPTY_DOUBLE_OBJECT_ARRAY;
  8479.         }
  8480.         return setAll(new Double[array.length], i -> Double.valueOf(array[i]));
  8481.     }

  8482.     /**
  8483.      * Converts an array of primitive floats to objects.
  8484.      *
  8485.      * <p>This method returns {@code null} for a {@code null} input array.</p>
  8486.      *
  8487.      * @param array  a {@code float} array
  8488.      * @return a {@link Float} array, {@code null} if null array input
  8489.      */
  8490.     public static Float[] toObject(final float[] array) {
  8491.         if (array == null) {
  8492.             return null;
  8493.         }
  8494.         if (array.length == 0) {
  8495.             return EMPTY_FLOAT_OBJECT_ARRAY;
  8496.         }
  8497.         return setAll(new Float[array.length], i -> Float.valueOf(array[i]));
  8498.     }

  8499.     /**
  8500.      * Converts an array of primitive ints to objects.
  8501.      *
  8502.      * <p>This method returns {@code null} for a {@code null} input array.</p>
  8503.      *
  8504.      * @param array  an {@code int} array
  8505.      * @return an {@link Integer} array, {@code null} if null array input
  8506.      */
  8507.     public static Integer[] toObject(final int[] array) {
  8508.         if (array == null) {
  8509.             return null;
  8510.         }
  8511.         if (array.length == 0) {
  8512.             return EMPTY_INTEGER_OBJECT_ARRAY;
  8513.         }
  8514.         return setAll(new Integer[array.length], i -> Integer.valueOf(array[i]));
  8515.     }

  8516.     /**
  8517.      * Converts an array of primitive longs to objects.
  8518.      *
  8519.      * <p>This method returns {@code null} for a {@code null} input array.</p>
  8520.      *
  8521.      * @param array  a {@code long} array
  8522.      * @return a {@link Long} array, {@code null} if null array input
  8523.      */
  8524.     public static Long[] toObject(final long[] array) {
  8525.         if (array == null) {
  8526.             return null;
  8527.         }
  8528.         if (array.length == 0) {
  8529.             return EMPTY_LONG_OBJECT_ARRAY;
  8530.         }
  8531.         return setAll(new Long[array.length], i -> Long.valueOf(array[i]));
  8532.     }

  8533.     /**
  8534.      * Converts an array of primitive shorts to objects.
  8535.      *
  8536.      * <p>This method returns {@code null} for a {@code null} input array.</p>
  8537.      *
  8538.      * @param array  a {@code short} array
  8539.      * @return a {@link Short} array, {@code null} if null array input
  8540.      */
  8541.     public static Short[] toObject(final short[] array) {
  8542.         if (array == null) {
  8543.             return null;
  8544.         }
  8545.         if (array.length == 0) {
  8546.             return EMPTY_SHORT_OBJECT_ARRAY;
  8547.         }
  8548.         return setAll(new Short[array.length], i -> Short.valueOf(array[i]));
  8549.     }

  8550.     /**
  8551.      * Converts an array of object Booleans to primitives.
  8552.      * <p>
  8553.      * This method returns {@code null} for a {@code null} input array.
  8554.      * </p>
  8555.      * <p>
  8556.      * Null array elements map to false, like {@code Boolean.parseBoolean(null)} and its callers return false.
  8557.      * </p>
  8558.      *
  8559.      * @param array a {@link Boolean} array, may be {@code null}
  8560.      * @return a {@code boolean} array, {@code null} if null array input
  8561.      */
  8562.     public static boolean[] toPrimitive(final Boolean[] array) {
  8563.         return toPrimitive(array, false);
  8564.     }

  8565.     /**
  8566.      * Converts an array of object Booleans to primitives handling {@code null}.
  8567.      * <p>
  8568.      * This method returns {@code null} for a {@code null} input array.
  8569.      * </p>
  8570.      *
  8571.      * @param array  a {@link Boolean} array, may be {@code null}
  8572.      * @param valueForNull  the value to insert if {@code null} found
  8573.      * @return a {@code boolean} array, {@code null} if null array input
  8574.      */
  8575.     public static boolean[] toPrimitive(final Boolean[] array, final boolean valueForNull) {
  8576.         if (array == null) {
  8577.             return null;
  8578.         }
  8579.         if (array.length == 0) {
  8580.             return EMPTY_BOOLEAN_ARRAY;
  8581.         }
  8582.         final boolean[] result = new boolean[array.length];
  8583.         for (int i = 0; i < array.length; i++) {
  8584.             final Boolean b = array[i];
  8585.             result[i] = b == null ? valueForNull : b.booleanValue();
  8586.         }
  8587.         return result;
  8588.     }

  8589.     /**
  8590.      * Converts an array of object Bytes to primitives.
  8591.      * <p>
  8592.      * This method returns {@code null} for a {@code null} input array.
  8593.      * </p>
  8594.      *
  8595.      * @param array  a {@link Byte} array, may be {@code null}
  8596.      * @return a {@code byte} array, {@code null} if null array input
  8597.      * @throws NullPointerException if an array element is {@code null}
  8598.      */
  8599.     public static byte[] toPrimitive(final Byte[] array) {
  8600.         if (array == null) {
  8601.             return null;
  8602.         }
  8603.         if (array.length == 0) {
  8604.             return EMPTY_BYTE_ARRAY;
  8605.         }
  8606.         final byte[] result = new byte[array.length];
  8607.         for (int i = 0; i < array.length; i++) {
  8608.             result[i] = array[i].byteValue();
  8609.         }
  8610.         return result;
  8611.     }

  8612.     /**
  8613.      * Converts an array of object Bytes to primitives handling {@code null}.
  8614.      * <p>
  8615.      * This method returns {@code null} for a {@code null} input array.
  8616.      * </p>
  8617.      *
  8618.      * @param array  a {@link Byte} array, may be {@code null}
  8619.      * @param valueForNull  the value to insert if {@code null} found
  8620.      * @return a {@code byte} array, {@code null} if null array input
  8621.      */
  8622.     public static byte[] toPrimitive(final Byte[] array, final byte valueForNull) {
  8623.         if (array == null) {
  8624.             return null;
  8625.         }
  8626.         if (array.length == 0) {
  8627.             return EMPTY_BYTE_ARRAY;
  8628.         }
  8629.         final byte[] result = new byte[array.length];
  8630.         for (int i = 0; i < array.length; i++) {
  8631.             final Byte b = array[i];
  8632.             result[i] = b == null ? valueForNull : b.byteValue();
  8633.         }
  8634.         return result;
  8635.     }

  8636.     /**
  8637.      * Converts an array of object Characters to primitives.
  8638.      * <p>
  8639.      * This method returns {@code null} for a {@code null} input array.
  8640.      * </p>
  8641.      *
  8642.      * @param array  a {@link Character} array, may be {@code null}
  8643.      * @return a {@code char} array, {@code null} if null array input
  8644.      * @throws NullPointerException if an array element is {@code null}
  8645.      */
  8646.     public static char[] toPrimitive(final Character[] array) {
  8647.         if (array == null) {
  8648.             return null;
  8649.         }
  8650.         if (array.length == 0) {
  8651.             return EMPTY_CHAR_ARRAY;
  8652.         }
  8653.         final char[] result = new char[array.length];
  8654.         for (int i = 0; i < array.length; i++) {
  8655.             result[i] = array[i].charValue();
  8656.         }
  8657.         return result;
  8658.     }

  8659.     /**
  8660.      * Converts an array of object Character to primitives handling {@code null}.
  8661.      * <p>
  8662.      * This method returns {@code null} for a {@code null} input array.
  8663.      * </p>
  8664.      *
  8665.      * @param array  a {@link Character} array, may be {@code null}
  8666.      * @param valueForNull  the value to insert if {@code null} found
  8667.      * @return a {@code char} array, {@code null} if null array input
  8668.      */
  8669.     public static char[] toPrimitive(final Character[] array, final char valueForNull) {
  8670.         if (array == null) {
  8671.             return null;
  8672.         }
  8673.         if (array.length == 0) {
  8674.             return EMPTY_CHAR_ARRAY;
  8675.         }
  8676.         final char[] result = new char[array.length];
  8677.         for (int i = 0; i < array.length; i++) {
  8678.             final Character b = array[i];
  8679.             result[i] = b == null ? valueForNull : b.charValue();
  8680.         }
  8681.         return result;
  8682.     }

  8683.     /**
  8684.      * Converts an array of object Doubles to primitives.
  8685.      * <p>
  8686.      * This method returns {@code null} for a {@code null} input array.
  8687.      * </p>
  8688.      *
  8689.      * @param array  a {@link Double} array, may be {@code null}
  8690.      * @return a {@code double} array, {@code null} if null array input
  8691.      * @throws NullPointerException if an array element is {@code null}
  8692.      */
  8693.     public static double[] toPrimitive(final Double[] array) {
  8694.         if (array == null) {
  8695.             return null;
  8696.         }
  8697.         if (array.length == 0) {
  8698.             return EMPTY_DOUBLE_ARRAY;
  8699.         }
  8700.         final double[] result = new double[array.length];
  8701.         for (int i = 0; i < array.length; i++) {
  8702.             result[i] = array[i].doubleValue();
  8703.         }
  8704.         return result;
  8705.     }

  8706.     /**
  8707.      * Converts an array of object Doubles to primitives handling {@code null}.
  8708.      * <p>
  8709.      * This method returns {@code null} for a {@code null} input array.
  8710.      * </p>
  8711.      *
  8712.      * @param array  a {@link Double} array, may be {@code null}
  8713.      * @param valueForNull  the value to insert if {@code null} found
  8714.      * @return a {@code double} array, {@code null} if null array input
  8715.      */
  8716.     public static double[] toPrimitive(final Double[] array, final double valueForNull) {
  8717.         if (array == null) {
  8718.             return null;
  8719.         }
  8720.         if (array.length == 0) {
  8721.             return EMPTY_DOUBLE_ARRAY;
  8722.         }
  8723.         final double[] result = new double[array.length];
  8724.         for (int i = 0; i < array.length; i++) {
  8725.             final Double b = array[i];
  8726.             result[i] = b == null ? valueForNull : b.doubleValue();
  8727.         }
  8728.         return result;
  8729.     }

  8730.     /**
  8731.      * Converts an array of object Floats to primitives.
  8732.      * <p>
  8733.      * This method returns {@code null} for a {@code null} input array.
  8734.      * </p>
  8735.      *
  8736.      * @param array  a {@link Float} array, may be {@code null}
  8737.      * @return a {@code float} array, {@code null} if null array input
  8738.      * @throws NullPointerException if an array element is {@code null}
  8739.      */
  8740.     public static float[] toPrimitive(final Float[] array) {
  8741.         if (array == null) {
  8742.             return null;
  8743.         }
  8744.         if (array.length == 0) {
  8745.             return EMPTY_FLOAT_ARRAY;
  8746.         }
  8747.         final float[] result = new float[array.length];
  8748.         for (int i = 0; i < array.length; i++) {
  8749.             result[i] = array[i].floatValue();
  8750.         }
  8751.         return result;
  8752.     }

  8753.     /**
  8754.      * Converts an array of object Floats to primitives handling {@code null}.
  8755.      * <p>
  8756.      * This method returns {@code null} for a {@code null} input array.
  8757.      * </p>
  8758.      *
  8759.      * @param array  a {@link Float} array, may be {@code null}
  8760.      * @param valueForNull  the value to insert if {@code null} found
  8761.      * @return a {@code float} array, {@code null} if null array input
  8762.      */
  8763.     public static float[] toPrimitive(final Float[] array, final float valueForNull) {
  8764.         if (array == null) {
  8765.             return null;
  8766.         }
  8767.         if (array.length == 0) {
  8768.             return EMPTY_FLOAT_ARRAY;
  8769.         }
  8770.         final float[] result = new float[array.length];
  8771.         for (int i = 0; i < array.length; i++) {
  8772.             final Float b = array[i];
  8773.             result[i] = b == null ? valueForNull : b.floatValue();
  8774.         }
  8775.         return result;
  8776.     }

  8777.     /**
  8778.      * Converts an array of object Integers to primitives.
  8779.      * <p>
  8780.      * This method returns {@code null} for a {@code null} input array.
  8781.      * </p>
  8782.      *
  8783.      * @param array  a {@link Integer} array, may be {@code null}
  8784.      * @return an {@code int} array, {@code null} if null array input
  8785.      * @throws NullPointerException if an array element is {@code null}
  8786.      */
  8787.     public static int[] toPrimitive(final Integer[] array) {
  8788.         if (array == null) {
  8789.             return null;
  8790.         }
  8791.         if (array.length == 0) {
  8792.             return EMPTY_INT_ARRAY;
  8793.         }
  8794.         final int[] result = new int[array.length];
  8795.         for (int i = 0; i < array.length; i++) {
  8796.             result[i] = array[i].intValue();
  8797.         }
  8798.         return result;
  8799.     }

  8800.     /**
  8801.      * Converts an array of object Integer to primitives handling {@code null}.
  8802.      * <p>
  8803.      * This method returns {@code null} for a {@code null} input array.
  8804.      * </p>
  8805.      *
  8806.      * @param array  a {@link Integer} array, may be {@code null}
  8807.      * @param valueForNull  the value to insert if {@code null} found
  8808.      * @return an {@code int} array, {@code null} if null array input
  8809.      */
  8810.     public static int[] toPrimitive(final Integer[] array, final int valueForNull) {
  8811.         if (array == null) {
  8812.             return null;
  8813.         }
  8814.         if (array.length == 0) {
  8815.             return EMPTY_INT_ARRAY;
  8816.         }
  8817.         final int[] result = new int[array.length];
  8818.         for (int i = 0; i < array.length; i++) {
  8819.             final Integer b = array[i];
  8820.             result[i] = b == null ? valueForNull : b.intValue();
  8821.         }
  8822.         return result;
  8823.     }

  8824.     /**
  8825.      * Converts an array of object Longs to primitives.
  8826.      * <p>
  8827.      * This method returns {@code null} for a {@code null} input array.
  8828.      * </p>
  8829.      *
  8830.      * @param array  a {@link Long} array, may be {@code null}
  8831.      * @return a {@code long} array, {@code null} if null array input
  8832.      * @throws NullPointerException if an array element is {@code null}
  8833.      */
  8834.     public static long[] toPrimitive(final Long[] array) {
  8835.         if (array == null) {
  8836.             return null;
  8837.         }
  8838.         if (array.length == 0) {
  8839.             return EMPTY_LONG_ARRAY;
  8840.         }
  8841.         final long[] result = new long[array.length];
  8842.         for (int i = 0; i < array.length; i++) {
  8843.             result[i] = array[i].longValue();
  8844.         }
  8845.         return result;
  8846.     }

  8847.     /**
  8848.      * Converts an array of object Long to primitives handling {@code null}.
  8849.      * <p>
  8850.      * This method returns {@code null} for a {@code null} input array.
  8851.      * </p>
  8852.      *
  8853.      * @param array  a {@link Long} array, may be {@code null}
  8854.      * @param valueForNull  the value to insert if {@code null} found
  8855.      * @return a {@code long} array, {@code null} if null array input
  8856.      */
  8857.     public static long[] toPrimitive(final Long[] array, final long valueForNull) {
  8858.         if (array == null) {
  8859.             return null;
  8860.         }
  8861.         if (array.length == 0) {
  8862.             return EMPTY_LONG_ARRAY;
  8863.         }
  8864.         final long[] result = new long[array.length];
  8865.         for (int i = 0; i < array.length; i++) {
  8866.             final Long b = array[i];
  8867.             result[i] = b == null ? valueForNull : b.longValue();
  8868.         }
  8869.         return result;
  8870.     }

  8871.     /**
  8872.      * Create an array of primitive type from an array of wrapper types.
  8873.      * <p>
  8874.      * This method returns {@code null} for a {@code null} input array.
  8875.      * </p>
  8876.      *
  8877.      * @param array  an array of wrapper object
  8878.      * @return an array of the corresponding primitive type, or the original array
  8879.      * @since 3.5
  8880.      */
  8881.     public static Object toPrimitive(final Object array) {
  8882.         if (array == null) {
  8883.             return null;
  8884.         }
  8885.         final Class<?> ct = array.getClass().getComponentType();
  8886.         final Class<?> pt = ClassUtils.wrapperToPrimitive(ct);
  8887.         if (Boolean.TYPE.equals(pt)) {
  8888.             return toPrimitive((Boolean[]) array);
  8889.         }
  8890.         if (Character.TYPE.equals(pt)) {
  8891.             return toPrimitive((Character[]) array);
  8892.         }
  8893.         if (Byte.TYPE.equals(pt)) {
  8894.             return toPrimitive((Byte[]) array);
  8895.         }
  8896.         if (Integer.TYPE.equals(pt)) {
  8897.             return toPrimitive((Integer[]) array);
  8898.         }
  8899.         if (Long.TYPE.equals(pt)) {
  8900.             return toPrimitive((Long[]) array);
  8901.         }
  8902.         if (Short.TYPE.equals(pt)) {
  8903.             return toPrimitive((Short[]) array);
  8904.         }
  8905.         if (Double.TYPE.equals(pt)) {
  8906.             return toPrimitive((Double[]) array);
  8907.         }
  8908.         if (Float.TYPE.equals(pt)) {
  8909.             return toPrimitive((Float[]) array);
  8910.         }
  8911.         return array;
  8912.     }

  8913.     /**
  8914.      * Converts an array of object Shorts to primitives.
  8915.      * <p>
  8916.      * This method returns {@code null} for a {@code null} input array.
  8917.      * </p>
  8918.      *
  8919.      * @param array  a {@link Short} array, may be {@code null}
  8920.      * @return a {@code byte} array, {@code null} if null array input
  8921.      * @throws NullPointerException if an array element is {@code null}
  8922.      */
  8923.     public static short[] toPrimitive(final Short[] array) {
  8924.         if (array == null) {
  8925.             return null;
  8926.         }
  8927.         if (array.length == 0) {
  8928.             return EMPTY_SHORT_ARRAY;
  8929.         }
  8930.         final short[] result = new short[array.length];
  8931.         for (int i = 0; i < array.length; i++) {
  8932.             result[i] = array[i].shortValue();
  8933.         }
  8934.         return result;
  8935.     }

  8936.     /**
  8937.      * Converts an array of object Short to primitives handling {@code null}.
  8938.      * <p>
  8939.      * This method returns {@code null} for a {@code null} input array.
  8940.      * </p>
  8941.      *
  8942.      * @param array  a {@link Short} array, may be {@code null}
  8943.      * @param valueForNull  the value to insert if {@code null} found
  8944.      * @return a {@code byte} array, {@code null} if null array input
  8945.      */
  8946.     public static short[] toPrimitive(final Short[] array, final short valueForNull) {
  8947.         if (array == null) {
  8948.             return null;
  8949.         }
  8950.         if (array.length == 0) {
  8951.             return EMPTY_SHORT_ARRAY;
  8952.         }
  8953.         final short[] result = new short[array.length];
  8954.         for (int i = 0; i < array.length; i++) {
  8955.             final Short b = array[i];
  8956.             result[i] = b == null ? valueForNull : b.shortValue();
  8957.         }
  8958.         return result;
  8959.     }

  8960.     /**
  8961.      * Outputs an array as a String, treating {@code null} as an empty array.
  8962.      * <p>
  8963.      * Multi-dimensional arrays are handled correctly, including
  8964.      * multi-dimensional primitive arrays.
  8965.      * </p>
  8966.      * <p>
  8967.      * The format is that of Java source code, for example {@code {a,b}}.
  8968.      * </p>
  8969.      *
  8970.      * @param array  the array to get a toString for, may be {@code null}
  8971.      * @return a String representation of the array, '{}' if null array input
  8972.      */
  8973.     public static String toString(final Object array) {
  8974.         return toString(array, "{}");
  8975.     }

  8976.     /**
  8977.      * Outputs an array as a String handling {@code null}s.
  8978.      * <p>
  8979.      * Multi-dimensional arrays are handled correctly, including
  8980.      * multi-dimensional primitive arrays.
  8981.      * </p>
  8982.      * <p>
  8983.      * The format is that of Java source code, for example {@code {a,b}}.
  8984.      * </p>
  8985.      *
  8986.      * @param array  the array to get a toString for, may be {@code null}
  8987.      * @param stringIfNull  the String to return if the array is {@code null}
  8988.      * @return a String representation of the array
  8989.      */
  8990.     public static String toString(final Object array, final String stringIfNull) {
  8991.         if (array == null) {
  8992.             return stringIfNull;
  8993.         }
  8994.         return new ToStringBuilder(array, ToStringStyle.SIMPLE_STYLE).append(array).toString();
  8995.     }

  8996.     /**
  8997.      * Returns an array containing the string representation of each element in the argument array.
  8998.      * <p>
  8999.      * This method returns {@code null} for a {@code null} input array.
  9000.      * </p>
  9001.      *
  9002.      * @param array the {@code Object[]} to be processed, may be null
  9003.      * @return {@code String[]} of the same size as the source with its element's string representation,
  9004.      * {@code null} if null array input
  9005.      * @throws NullPointerException if an array element is {@code null}
  9006.      * @since 3.6
  9007.      */
  9008.     public static String[] toStringArray(final Object[] array) {
  9009.         if (array == null) {
  9010.             return null;
  9011.         }
  9012.         if (array.length == 0) {
  9013.             return EMPTY_STRING_ARRAY;
  9014.         }

  9015.         final String[] result = new String[array.length];
  9016.         for (int i = 0; i < array.length; i++) {
  9017.             result[i] = array[i].toString();
  9018.         }

  9019.         return result;
  9020.     }

  9021.     /**
  9022.      * Returns an array containing the string representation of each element in the argument
  9023.      * array handling {@code null} elements.
  9024.      * <p>
  9025.      * This method returns {@code null} for a {@code null} input array.
  9026.      * </p>
  9027.      *
  9028.      * @param array the Object[] to be processed, may be null
  9029.      * @param valueForNullElements the value to insert if {@code null} is found
  9030.      * @return a {@link String} array, {@code null} if null array input
  9031.      * @since 3.6
  9032.      */
  9033.     public static String[] toStringArray(final Object[] array, final String valueForNullElements) {
  9034.         if (null == array) {
  9035.             return null;
  9036.         }
  9037.         if (array.length == 0) {
  9038.             return EMPTY_STRING_ARRAY;
  9039.         }

  9040.         final String[] result = new String[array.length];
  9041.         for (int i = 0; i < array.length; i++) {
  9042.             result[i] = Objects.toString(array[i], valueForNullElements);
  9043.         }

  9044.         return result;
  9045.     }

  9046.     /**
  9047.      * ArrayUtils instances should NOT be constructed in standard programming.
  9048.      * Instead, the class should be used as {@code ArrayUtils.clone(new int[] {2})}.
  9049.      * <p>
  9050.      * This constructor is public to permit tools that require a JavaBean instance
  9051.      * to operate.
  9052.      * </p>
  9053.      *
  9054.      * @deprecated TODO Make private in 4.0.
  9055.      */
  9056.     @Deprecated
  9057.     public ArrayUtils() {
  9058.         // empty
  9059.     }
  9060. }