ComparableUtils.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.compare;

  18. import java.util.function.Predicate;

  19. import org.apache.commons.lang3.ObjectUtils;

  20. /**
  21.  * Utility library to provide helper methods for translating {@link Comparable#compareTo} result into a boolean.
  22.  *
  23.  * <p>Example: {@code boolean x = is(myComparable).lessThanOrEqualTo(otherComparable)}</p>
  24.  *
  25.  * <p>#ThreadSafe#</p>
  26.  *
  27.  * @since 3.10
  28.  */
  29. public class ComparableUtils {

  30.     /**
  31.      * Provides access to the available methods
  32.      *
  33.      * @param <A> the type of objects that this object may be compared against.
  34.      */
  35.     public static class ComparableCheckBuilder<A extends Comparable<A>> {

  36.         private final A a;

  37.         private ComparableCheckBuilder(final A a) {
  38.             this.a = a;
  39.         }

  40.         /**
  41.          * Checks if {@code [b <= a <= c]} or {@code [b >= a >= c]} where the {@code a} is object passed to {@link #is}.
  42.          *
  43.          * @param b the object to compare to the base object
  44.          * @param c the object to compare to the base object
  45.          * @return true if the base object is between b and c
  46.          */
  47.         public boolean between(final A b, final A c) {
  48.             return betweenOrdered(b, c) || betweenOrdered(c, b);
  49.         }

  50.         /**
  51.          * Checks if {@code (b < a < c)} or {@code (b > a > c)} where the {@code a} is object passed to {@link #is}.
  52.          *
  53.          * @param b the object to compare to the base object
  54.          * @param c the object to compare to the base object
  55.          * @return true if the base object is between b and c and not equal to those
  56.          */
  57.         public boolean betweenExclusive(final A b, final A c) {
  58.             return betweenOrderedExclusive(b, c) || betweenOrderedExclusive(c, b);
  59.         }

  60.         private boolean betweenOrdered(final A b, final A c) {
  61.             return greaterThanOrEqualTo(b) && lessThanOrEqualTo(c);
  62.         }

  63.         private boolean betweenOrderedExclusive(final A b, final A c) {
  64.             return greaterThan(b) && lessThan(c);
  65.         }

  66.         /**
  67.          * Checks if the object passed to {@link #is} is equal to {@code b}
  68.          *
  69.          * @param b the object to compare to the base object
  70.          * @return true if the value returned by {@link Comparable#compareTo} is equal to {@code 0}
  71.          */
  72.         public boolean equalTo(final A b) {
  73.             return a.compareTo(b) == 0;
  74.         }

  75.         /**
  76.          * Checks if the object passed to {@link #is} is greater than {@code b}
  77.          *
  78.          * @param b the object to compare to the base object
  79.          * @return true if the value returned by {@link Comparable#compareTo} is greater than {@code 0}
  80.          */
  81.         public boolean greaterThan(final A b) {
  82.             return a.compareTo(b) > 0;
  83.         }

  84.         /**
  85.          * Checks if the object passed to {@link #is} is greater than or equal to {@code b}
  86.          *
  87.          * @param b the object to compare to the base object
  88.          * @return true if the value returned by {@link Comparable#compareTo} is greater than or equal to {@code 0}
  89.          */
  90.         public boolean greaterThanOrEqualTo(final A b) {
  91.             return a.compareTo(b) >= 0;
  92.         }

  93.         /**
  94.          * Checks if the object passed to {@link #is} is less than {@code b}
  95.          *
  96.          * @param b the object to compare to the base object
  97.          * @return true if the value returned by {@link Comparable#compareTo} is less than {@code 0}
  98.          */
  99.         public boolean lessThan(final A b) {
  100.             return a.compareTo(b) < 0;
  101.         }

  102.         /**
  103.          * Checks if the object passed to {@link #is} is less than or equal to {@code b}
  104.          *
  105.          * @param b the object to compare to the base object
  106.          * @return true if the value returned by {@link Comparable#compareTo} is less than or equal to {@code 0}
  107.          */
  108.         public boolean lessThanOrEqualTo(final A b) {
  109.             return a.compareTo(b) <= 0;
  110.         }
  111.     }

  112.     /**
  113.      * Checks if {@code [b <= a <= c]} or {@code [b >= a >= c]} where the {@code a} is the tested object.
  114.      *
  115.      * @param b the object to compare to the tested object
  116.      * @param c the object to compare to the tested object
  117.      * @param <A> type of the test object
  118.      * @return a predicate for true if the tested object is between b and c
  119.      */
  120.     public static <A extends Comparable<A>> Predicate<A> between(final A b, final A c) {
  121.         return a -> is(a).between(b, c);
  122.     }

  123.     /**
  124.      * Checks if {@code (b < a < c)} or {@code (b > a > c)} where the {@code a} is the tested object.
  125.      *
  126.      * @param b the object to compare to the tested object
  127.      * @param c the object to compare to the tested object
  128.      * @param <A> type of the test object
  129.      * @return a predicate for true if the tested object is between b and c and not equal to those
  130.      */
  131.     public static <A extends Comparable<A>> Predicate<A> betweenExclusive(final A b, final A c) {
  132.         return a -> is(a).betweenExclusive(b, c);
  133.     }

  134.     /**
  135.      * Checks if the tested object is greater than or equal to {@code b}
  136.      *
  137.      * @param b the object to compare to the tested object
  138.      * @param <A> type of the test object
  139.      * @return a predicate for true if the value returned by {@link Comparable#compareTo}
  140.      * is greater than or equal to {@code 0}
  141.      */
  142.     public static <A extends Comparable<A>> Predicate<A> ge(final A b) {
  143.         return a -> is(a).greaterThanOrEqualTo(b);
  144.     }

  145.     /**
  146.      * Checks if the tested object is greater than {@code b}
  147.      *
  148.      * @param b the object to compare to the tested object
  149.      * @param <A> type of the test object
  150.      * @return a predicate for true if the value returned by {@link Comparable#compareTo} is greater than {@code 0}
  151.      */
  152.     public static <A extends Comparable<A>> Predicate<A> gt(final A b) {
  153.         return a -> is(a).greaterThan(b);
  154.     }

  155.     /**
  156.      * Provides access to the available methods
  157.      *
  158.      * @param a base object in the further comparison
  159.      * @param <A> type of the base object
  160.      * @return a builder object with further methods
  161.      */
  162.     public static <A extends Comparable<A>> ComparableCheckBuilder<A> is(final A a) {
  163.         return new ComparableCheckBuilder<>(a);
  164.     }

  165.     /**
  166.      * Checks if the tested object is less than or equal to {@code b}
  167.      *
  168.      * @param b the object to compare to the tested object
  169.      * @param <A> type of the test object
  170.      * @return a predicate for true if the value returned by {@link Comparable#compareTo}
  171.      * is less than or equal to {@code 0}
  172.      */
  173.     public static <A extends Comparable<A>> Predicate<A> le(final A b) {
  174.         return a -> is(a).lessThanOrEqualTo(b);
  175.     }

  176.     /**
  177.      * Checks if the tested object is less than {@code b}
  178.      *
  179.      * @param b the object to compare to the tested object
  180.      * @param <A> type of the test object
  181.      * @return a predicate for true if the value returned by {@link Comparable#compareTo} is less than {@code 0}
  182.      */
  183.     public static <A extends Comparable<A>> Predicate<A> lt(final A b) {
  184.         return a -> is(a).lessThan(b);
  185.     }

  186.     /**
  187.      * Returns the greater of two {@link Comparable} values, ignoring null.
  188.      * <p>
  189.      * For three or more values, use {@link ObjectUtils#max(Comparable...)}.
  190.      * </p>
  191.      *
  192.      * @param <A> Type of what we are comparing.
  193.      * @param comparable1 the first comparable, may be null.
  194.      * @param comparable2 the second comparable, may be null.
  195.      * @return the largest of {@code comparable1} and {@code comparable2}.
  196.      * @see ObjectUtils#max(Comparable...)
  197.      * @since 3.13.0
  198.      */
  199.     public static <A extends Comparable<A>> A max(final A comparable1, final A comparable2) {
  200.         return ObjectUtils.compare(comparable1, comparable2, false) > 0 ? comparable1 : comparable2;
  201.     }

  202.     /**
  203.      * Returns the lesser of two {@link Comparable} values, ignoring null.
  204.      * <p>
  205.      * For three or more values, use {@link ObjectUtils#min(Comparable...)}.
  206.      * </p>
  207.      *
  208.      * @param <A> Type of what we are comparing.
  209.      * @param comparable1 the first comparable, may be null.
  210.      * @param comparable2 the second comparable, may be null.
  211.      * @return the smallest of {@code comparable1} and {@code comparable2}.
  212.      * @see ObjectUtils#min(Comparable...)
  213.      * @since 3.13.0
  214.      */
  215.     public static <A extends Comparable<A>> A min(final A comparable1, final A comparable2) {
  216.         return ObjectUtils.compare(comparable1, comparable2, true) < 0 ? comparable1 : comparable2;
  217.     }

  218.     private ComparableUtils() {
  219.         // empty
  220.     }
  221. }