Range.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.io.Serializable;
  19. import java.util.Comparator;
  20. import java.util.Objects;

  21. /**
  22.  * An immutable range of objects from a minimum to maximum point inclusive.
  23.  *
  24.  * <p>The objects need to either be implementations of {@link Comparable}
  25.  * or you need to supply a {@link Comparator}.</p>
  26.  *
  27.  * <p>#ThreadSafe# if the objects and comparator are thread-safe.</p>
  28.  *
  29.  * @param <T> The type of range values.
  30.  * @since 3.0
  31.  */
  32. public class Range<T> implements Serializable {

  33.     @SuppressWarnings({"rawtypes", "unchecked"})
  34.     private enum ComparableComparator implements Comparator {
  35.         INSTANCE;

  36.         /**
  37.          * Comparable based compare implementation.
  38.          *
  39.          * @param obj1 left-hand side side of comparison
  40.          * @param obj2 right-hand side side of comparison
  41.          * @return negative, 0, positive comparison value
  42.          */
  43.         @Override
  44.         public int compare(final Object obj1, final Object obj2) {
  45.             return ((Comparable) obj1).compareTo(obj2);
  46.         }
  47.     }

  48.     /**
  49.      * Serialization version.
  50.      *
  51.      * @see java.io.Serializable
  52.      */
  53.     private static final long serialVersionUID = 1L;

  54.     /**
  55.      * Creates a range with the specified minimum and maximum values (both inclusive).
  56.      *
  57.      * <p>The range uses the natural ordering of the elements to determine where
  58.      * values lie in the range.</p>
  59.      *
  60.      * <p>The arguments may be passed in the order (min,max) or (max,min).
  61.      * The getMinimum and getMaximum methods will return the correct values.</p>
  62.      *
  63.      * @param <T> the type of the elements in this range
  64.      * @param fromInclusive  the first value that defines the edge of the range, inclusive
  65.      * @param toInclusive  the second value that defines the edge of the range, inclusive
  66.      * @return the range object, not null
  67.      * @throws NullPointerException when fromInclusive is null.
  68.      * @throws NullPointerException when toInclusive is null.
  69.      * @throws ClassCastException if the elements are not {@link Comparable}
  70.      * @deprecated Use {@link #of(Comparable, Comparable)}.
  71.      */
  72.     @Deprecated
  73.     public static <T extends Comparable<? super T>> Range<T> between(final T fromInclusive, final T toInclusive) {
  74.         return of(fromInclusive, toInclusive, null);
  75.     }

  76.     /**
  77.      * Creates a range with the specified minimum and maximum values (both inclusive).
  78.      *
  79.      * <p>The range uses the specified {@link Comparator} to determine where
  80.      * values lie in the range.</p>
  81.      *
  82.      * <p>The arguments may be passed in the order (min,max) or (max,min).
  83.      * The getMinimum and getMaximum methods will return the correct values.</p>
  84.      *
  85.      * @param <T> the type of the elements in this range
  86.      * @param fromInclusive  the first value that defines the edge of the range, inclusive
  87.      * @param toInclusive  the second value that defines the edge of the range, inclusive
  88.      * @param comparator  the comparator to be used, null for natural ordering
  89.      * @return the range object, not null
  90.      * @throws NullPointerException when fromInclusive is null.
  91.      * @throws NullPointerException when toInclusive is null.
  92.      * @throws ClassCastException if using natural ordering and the elements are not {@link Comparable}
  93.      * @deprecated Use {@link #of(Object, Object, Comparator)}.
  94.      */
  95.     @Deprecated
  96.     public static <T> Range<T> between(final T fromInclusive, final T toInclusive, final Comparator<T> comparator) {
  97.         return new Range<>(fromInclusive, toInclusive, comparator);
  98.     }

  99.     /**
  100.      * Creates a range using the specified element as both the minimum
  101.      * and maximum in this range.
  102.      *
  103.      * <p>The range uses the natural ordering of the elements to determine where
  104.      * values lie in the range.</p>
  105.      *
  106.      * @param <T> the type of the elements in this range
  107.      * @param element  the value to use for this range, not null
  108.      * @return the range object, not null
  109.      * @throws NullPointerException if the element is null
  110.      * @throws ClassCastException if the element is not {@link Comparable}
  111.      */
  112.     public static <T extends Comparable<? super T>> Range<T> is(final T element) {
  113.         return of(element, element, null);
  114.     }

  115.     /**
  116.      * Creates a range using the specified element as both the minimum
  117.      * and maximum in this range.
  118.      *
  119.      * <p>The range uses the specified {@link Comparator} to determine where
  120.      * values lie in the range.</p>
  121.      *
  122.      * @param <T> the type of the elements in this range
  123.      * @param element  the value to use for this range, must not be {@code null}
  124.      * @param comparator  the comparator to be used, null for natural ordering
  125.      * @return the range object, not null
  126.      * @throws NullPointerException if the element is null
  127.      * @throws ClassCastException if using natural ordering and the elements are not {@link Comparable}
  128.      */
  129.     public static <T> Range<T> is(final T element, final Comparator<T> comparator) {
  130.         return of(element, element, comparator);
  131.     }

  132.     /**
  133.      * Creates a range with the specified minimum and maximum values (both inclusive).
  134.      *
  135.      * <p>The range uses the natural ordering of the elements to determine where
  136.      * values lie in the range.</p>
  137.      *
  138.      * <p>The arguments may be passed in the order (min,max) or (max,min).
  139.      * The getMinimum and getMaximum methods will return the correct values.</p>
  140.      *
  141.      * @param <T> the type of the elements in this range
  142.      * @param fromInclusive  the first value that defines the edge of the range, inclusive
  143.      * @param toInclusive  the second value that defines the edge of the range, inclusive
  144.      * @return the range object, not null
  145.      * @throws NullPointerException if either element is null
  146.      * @throws ClassCastException if the elements are not {@link Comparable}
  147.      * @since 3.13.0
  148.      */
  149.     public static <T extends Comparable<? super T>> Range<T> of(final T fromInclusive, final T toInclusive) {
  150.         return of(fromInclusive, toInclusive, null);
  151.     }

  152.     /**
  153.      * Creates a range with the specified minimum and maximum values (both inclusive).
  154.      *
  155.      * <p>The range uses the specified {@link Comparator} to determine where
  156.      * values lie in the range.</p>
  157.      *
  158.      * <p>The arguments may be passed in the order (min,max) or (max,min).
  159.      * The getMinimum and getMaximum methods will return the correct values.</p>
  160.      *
  161.      * @param <T> the type of the elements in this range
  162.      * @param fromInclusive  the first value that defines the edge of the range, inclusive
  163.      * @param toInclusive  the second value that defines the edge of the range, inclusive
  164.      * @param comparator  the comparator to be used, null for natural ordering
  165.      * @return the range object, not null
  166.      * @throws NullPointerException when fromInclusive is null.
  167.      * @throws NullPointerException when toInclusive is null.
  168.      * @throws ClassCastException if using natural ordering and the elements are not {@link Comparable}
  169.      * @since 3.13.0
  170.      */
  171.     public static <T> Range<T> of(final T fromInclusive, final T toInclusive, final Comparator<T> comparator) {
  172.         return new Range<>(fromInclusive, toInclusive, comparator);
  173.     }

  174.     /**
  175.      * The ordering scheme used in this range.
  176.      */
  177.     private final Comparator<T> comparator;

  178.     /**
  179.      * Cached output hashCode (class is immutable).
  180.      */
  181.     private transient int hashCode;

  182.     /**
  183.      * The maximum value in this range (inclusive).
  184.      */
  185.     private final T maximum;

  186.     /**
  187.      * The minimum value in this range (inclusive).
  188.      */
  189.     private final T minimum;

  190.     /**
  191.      * Cached output toString (class is immutable).
  192.      */
  193.     private transient String toString;

  194.     /**
  195.      * Creates an instance.
  196.      *
  197.      * @param element1  the first element, not null
  198.      * @param element2  the second element, not null
  199.      * @param comp  the comparator to be used, null for natural ordering
  200.      * @throws NullPointerException when element1 is null.
  201.      * @throws NullPointerException when element2 is null.
  202.      */
  203.     @SuppressWarnings("unchecked")
  204.     Range(final T element1, final T element2, final Comparator<T> comp) {
  205.         Objects.requireNonNull(element1, "element1");
  206.         Objects.requireNonNull(element2, "element2");
  207.         if (comp == null) {
  208.             this.comparator = ComparableComparator.INSTANCE;
  209.         } else {
  210.             this.comparator = comp;
  211.         }
  212.         if (this.comparator.compare(element1, element2) < 1) {
  213.             this.minimum = element1;
  214.             this.maximum = element2;
  215.         } else {
  216.             this.minimum = element2;
  217.             this.maximum = element1;
  218.         }
  219.     }

  220.     /**
  221.      * Checks whether the specified element occurs within this range.
  222.      *
  223.      * @param element  the element to check for, null returns false
  224.      * @return true if the specified element occurs within this range
  225.      */
  226.     public boolean contains(final T element) {
  227.         if (element == null) {
  228.             return false;
  229.         }
  230.         return comparator.compare(element, minimum) > -1 && comparator.compare(element, maximum) < 1;
  231.     }

  232.     /**
  233.      * Checks whether this range contains all the elements of the specified range.
  234.      *
  235.      * <p>This method may fail if the ranges have two different comparators or element types.</p>
  236.      *
  237.      * @param otherRange  the range to check, null returns false
  238.      * @return true if this range contains the specified range
  239.      * @throws RuntimeException if ranges cannot be compared
  240.      */
  241.     public boolean containsRange(final Range<T> otherRange) {
  242.         if (otherRange == null) {
  243.             return false;
  244.         }
  245.         return contains(otherRange.minimum)
  246.             && contains(otherRange.maximum);
  247.     }

  248.     /**
  249.      * Checks where the specified element occurs relative to this range.
  250.      *
  251.      * <p>The API is reminiscent of the Comparable interface returning {@code -1} if
  252.      * the element is before the range, {@code 0} if contained within the range and
  253.      * {@code 1} if the element is after the range.</p>
  254.      *
  255.      * @param element  the element to check for, not null
  256.      * @return -1, 0 or +1 depending on the element's location relative to the range
  257.      * @throws NullPointerException if {@code element} is {@code null}
  258.      */
  259.     public int elementCompareTo(final T element) {
  260.         // Comparable API says throw NPE on null
  261.         Objects.requireNonNull(element, "element");
  262.         if (isAfter(element)) {
  263.             return -1;
  264.         }
  265.         if (isBefore(element)) {
  266.             return 1;
  267.         }
  268.         return 0;
  269.     }

  270.     /**
  271.      * Compares this range to another object to test if they are equal.
  272.      *
  273.      * <p>To be equal, the minimum and maximum values must be equal, which
  274.      * ignores any differences in the comparator.</p>
  275.      *
  276.      * @param obj the reference object with which to compare
  277.      * @return true if this object is equal
  278.      */
  279.     @Override
  280.     public boolean equals(final Object obj) {
  281.         if (obj == this) {
  282.             return true;
  283.         }
  284.         if (obj == null || obj.getClass() != getClass()) {
  285.             return false;
  286.         }
  287.         @SuppressWarnings("unchecked") // OK because we checked the class above
  288.         final
  289.         Range<T> range = (Range<T>) obj;
  290.         return minimum.equals(range.minimum) &&
  291.                maximum.equals(range.maximum);
  292.     }

  293.     /**
  294.      * Fits the given element into this range by returning the given element or, if out of bounds, the range minimum if
  295.      * below, or the range maximum if above.
  296.      *
  297.      * <pre>{@code
  298.      * Range<Integer> range = Range.between(16, 64);
  299.      * range.fit(-9) -->  16
  300.      * range.fit(0)  -->  16
  301.      * range.fit(15) -->  16
  302.      * range.fit(16) -->  16
  303.      * range.fit(17) -->  17
  304.      * ...
  305.      * range.fit(63) -->  63
  306.      * range.fit(64) -->  64
  307.      * range.fit(99) -->  64
  308.      * }</pre>
  309.      * @param element the element to check for, not null
  310.      * @return the minimum, the element, or the maximum depending on the element's location relative to the range
  311.      * @throws NullPointerException if {@code element} is {@code null}
  312.      * @since 3.10
  313.      */
  314.     public T fit(final T element) {
  315.         // Comparable API says throw NPE on null
  316.         Objects.requireNonNull(element, "element");
  317.         if (isAfter(element)) {
  318.             return minimum;
  319.         }
  320.         if (isBefore(element)) {
  321.             return maximum;
  322.         }
  323.         return element;
  324.     }

  325.     /**
  326.      * Gets the comparator being used to determine if objects are within the range.
  327.      *
  328.      * <p>Natural ordering uses an internal comparator implementation, thus this
  329.      * method never returns null. See {@link #isNaturalOrdering()}.</p>
  330.      *
  331.      * @return the comparator being used, not null
  332.      */
  333.     public Comparator<T> getComparator() {
  334.         return comparator;
  335.     }

  336.     /**
  337.      * Gets the maximum value in this range.
  338.      *
  339.      * @return the maximum value in this range, not null
  340.      */
  341.     public T getMaximum() {
  342.         return maximum;
  343.     }

  344.     /**
  345.      * Gets the minimum value in this range.
  346.      *
  347.      * @return the minimum value in this range, not null
  348.      */
  349.     public T getMinimum() {
  350.         return minimum;
  351.     }

  352.     /**
  353.      * Gets a suitable hash code for the range.
  354.      *
  355.      * @return a hash code value for this object
  356.      */
  357.     @Override
  358.     public int hashCode() {
  359.         int result = hashCode;
  360.         if (hashCode == 0) {
  361.             result = 17;
  362.             result = 37 * result + getClass().hashCode();
  363.             result = 37 * result + minimum.hashCode();
  364.             result = 37 * result + maximum.hashCode();
  365.             hashCode = result;
  366.         }
  367.         return result;
  368.     }

  369.     /**
  370.      * Calculate the intersection of {@code this} and an overlapping Range.
  371.      * @param other overlapping Range
  372.      * @return range representing the intersection of {@code this} and {@code other} ({@code this} if equal)
  373.      * @throws IllegalArgumentException if {@code other} does not overlap {@code this}
  374.      * @since 3.0.1
  375.      */
  376.     public Range<T> intersectionWith(final Range<T> other) {
  377.         if (!this.isOverlappedBy(other)) {
  378.             throw new IllegalArgumentException(String.format(
  379.                 "Cannot calculate intersection with non-overlapping range %s", other));
  380.         }
  381.         if (this.equals(other)) {
  382.             return this;
  383.         }
  384.         final T min = getComparator().compare(minimum, other.minimum) < 0 ? other.minimum : minimum;
  385.         final T max = getComparator().compare(maximum, other.maximum) < 0 ? maximum : other.maximum;
  386.         return of(min, max, getComparator());
  387.     }

  388.     /**
  389.      * Checks whether this range is after the specified element.
  390.      *
  391.      * @param element  the element to check for, null returns false
  392.      * @return true if this range is entirely after the specified element
  393.      */
  394.     public boolean isAfter(final T element) {
  395.         if (element == null) {
  396.             return false;
  397.         }
  398.         return comparator.compare(element, minimum) < 0;
  399.     }

  400.     /**
  401.      * Checks whether this range is completely after the specified range.
  402.      *
  403.      * <p>This method may fail if the ranges have two different comparators or element types.</p>
  404.      *
  405.      * @param otherRange  the range to check, null returns false
  406.      * @return true if this range is completely after the specified range
  407.      * @throws RuntimeException if ranges cannot be compared
  408.      */
  409.     public boolean isAfterRange(final Range<T> otherRange) {
  410.         if (otherRange == null) {
  411.             return false;
  412.         }
  413.         return isAfter(otherRange.maximum);
  414.     }

  415.     /**
  416.      * Checks whether this range is before the specified element.
  417.      *
  418.      * @param element  the element to check for, null returns false
  419.      * @return true if this range is entirely before the specified element
  420.      */
  421.     public boolean isBefore(final T element) {
  422.         if (element == null) {
  423.             return false;
  424.         }
  425.         return comparator.compare(element, maximum) > 0;
  426.     }

  427.     /**
  428.      * Checks whether this range is completely before the specified range.
  429.      *
  430.      * <p>This method may fail if the ranges have two different comparators or element types.</p>
  431.      *
  432.      * @param otherRange  the range to check, null returns false
  433.      * @return true if this range is completely before the specified range
  434.      * @throws RuntimeException if ranges cannot be compared
  435.      */
  436.     public boolean isBeforeRange(final Range<T> otherRange) {
  437.         if (otherRange == null) {
  438.             return false;
  439.         }
  440.         return isBefore(otherRange.minimum);
  441.     }

  442.     /**
  443.      * Checks whether this range ends with the specified element.
  444.      *
  445.      * @param element  the element to check for, null returns false
  446.      * @return true if the specified element occurs within this range
  447.      */
  448.     public boolean isEndedBy(final T element) {
  449.         if (element == null) {
  450.             return false;
  451.         }
  452.         return comparator.compare(element, maximum) == 0;
  453.     }

  454.     /**
  455.      * Whether or not the Range is using the natural ordering of the elements.
  456.      *
  457.      * <p>Natural ordering uses an internal comparator implementation, thus this
  458.      * method is the only way to check if a null comparator was specified.</p>
  459.      *
  460.      * @return true if using natural ordering
  461.      */
  462.     public boolean isNaturalOrdering() {
  463.         return comparator == ComparableComparator.INSTANCE;
  464.     }

  465.     /**
  466.      * Checks whether this range is overlapped by the specified range.
  467.      *
  468.      * <p>Two ranges overlap if there is at least one element in common.</p>
  469.      *
  470.      * <p>This method may fail if the ranges have two different comparators or element types.</p>
  471.      *
  472.      * @param otherRange  the range to test, null returns false
  473.      * @return true if the specified range overlaps with this
  474.      *  range; otherwise, {@code false}
  475.      * @throws RuntimeException if ranges cannot be compared
  476.      */
  477.     public boolean isOverlappedBy(final Range<T> otherRange) {
  478.         if (otherRange == null) {
  479.             return false;
  480.         }
  481.         return otherRange.contains(minimum)
  482.             || otherRange.contains(maximum)
  483.             || contains(otherRange.minimum);
  484.     }

  485.     /**
  486.      * Checks whether this range starts with the specified element.
  487.      *
  488.      * @param element  the element to check for, null returns false
  489.      * @return true if the specified element occurs within this range
  490.      */
  491.     public boolean isStartedBy(final T element) {
  492.         if (element == null) {
  493.             return false;
  494.         }
  495.         return comparator.compare(element, minimum) == 0;
  496.     }

  497.     /**
  498.      * Gets the range as a {@link String}.
  499.      *
  500.      * <p>The format of the String is '[<em>min</em>..<em>max</em>]'.</p>
  501.      *
  502.      * @return the {@link String} representation of this range
  503.      */
  504.     @Override
  505.     public String toString() {
  506.         if (toString == null) {
  507.             toString = "[" + minimum + ".." + maximum + "]";
  508.         }
  509.         return toString;
  510.     }

  511.     /**
  512.      * Formats the receiver using the given format.
  513.      *
  514.      * <p>This uses {@link java.util.Formattable} to perform the formatting. Three variables may
  515.      * be used to embed the minimum, maximum and comparator.
  516.      * Use {@code %1$s} for the minimum element, {@code %2$s} for the maximum element
  517.      * and {@code %3$s} for the comparator.
  518.      * The default format used by {@code toString()} is {@code [%1$s..%2$s]}.</p>
  519.      *
  520.      * @param format  the format string, optionally containing {@code %1$s}, {@code %2$s} and  {@code %3$s}, not null
  521.      * @return the formatted string, not null
  522.      */
  523.     public String toString(final String format) {
  524.         return String.format(format, minimum, maximum, comparator);
  525.     }

  526. }