package fj; import fj.data.Array; import fj.data.Either; import fj.data.List; import fj.data.Natural; import fj.data.NonEmptyList; import fj.data.Option; import fj.data.Set; import fj.data.Stream; import fj.data.Validation; import java.math.BigDecimal; import java.math.BigInteger; import static fj.Function.curry; /** * Tests for ordering between two objects. * * @version %build.number% */ public final class Ord { private final F> f; private Ord(final F> f) { this.f = f; } /** * First-class ordering. * * @return A function that returns an ordering for its arguments. */ public F> compare() { return f; } /** * Returns an ordering for the given arguments. * * @param a1 An instance to compare for ordering to another. * @param a2 An instance to compare for ordering to another. * @return An ordering for the given arguments. */ public Ordering compare(final A a1, final A a2) { return f.f(a1).f(a2); } /** * Returns true if the given arguments are equal, false otherwise. * * @param a1 An instance to compare for equality to another. * @param a2 An instance to compare for equality to another. * @return true if the given arguments are equal, false otherwise. */ public boolean eq(final A a1, final A a2) { return compare(a1, a2) == Ordering.EQ; } /** * Returns an Equal for this order. * * @return An Equal for this order. */ public Equal equal() { return Equal.equal(curry(new F2() { public Boolean f(final A a1, final A a2) { return eq(a1, a2); } })); } /** * Maps the given function across this ord as a contra-variant functor. * * @param f The function to map. * @return A new ord. */ public Ord comap(final F f) { return ord(f.andThen().o(this.f).o(f)); } /** * Returns true if the first given argument is less than the second given argument, * false otherwise. * * @param a1 An instance to compare for ordering to another. * @param a2 An instance to compare for ordering to another. * @return true if the first given argument is less than the second given argument, * false otherwise. */ public boolean isLessThan(final A a1, final A a2) { return compare(a1, a2) == Ordering.LT; } /** * Returns true if the first given argument is greater than the second given * argument, false otherwise. * * @param a1 An instance to compare for ordering to another. * @param a2 An instance to compare for ordering to another. * @return true if the first given argument is greater than the second given * argument, false otherwise. */ public boolean isGreaterThan(final A a1, final A a2) { return compare(a1, a2) == Ordering.GT; } /** * Returns a function that returns true if its argument is less than the argument to this method. * * @param a A value to compare against. * @return A function that returns true if its argument is less than the argument to this method. */ public F isLessThan(final A a) { return new F() { public Boolean f(final A a2) { return compare(a2, a) == Ordering.LT; } }; } /** * Returns a function that returns true if its argument is greater than than the argument to this method. * * @param a A value to compare against. * @return A function that returns true if its argument is greater than the argument to this method. */ public F isGreaterThan(final A a) { return new F() { public Boolean f(final A a2) { return compare(a2, a) == Ordering.GT; } }; } /** * Returns the greater of its two arguments. * * @param a1 A value to compare with another. * @param a2 A value to compare with another. * @return The greater of the two values. */ public A max(final A a1, final A a2) { return isGreaterThan(a1, a2) ? a1 : a2; } /** * Returns the lesser of its two arguments. * * @param a1 A value to compare with another. * @param a2 A value to compare with another. * @return The lesser of the two values. */ public A min(final A a1, final A a2) { return isLessThan(a1, a2) ? a1 : a2; } /** * A function that returns the greater of its two arguments. */ public final F> max = curry(new F2() { public A f(final A a, final A a1) { return max(a, a1); } }); /** * A function that returns the lesser of its two arguments. */ public final F> min = curry(new F2() { public A f(final A a, final A a1) { return min(a, a1); } }); /** * Returns an order instance that uses the given equality test and ordering function. * * @param f The order function. * @return An order instance. */ public static Ord ord(final F> f) { return new Ord(f); } /** * An order instance for the boolean type. */ public static final Ord booleanOrd = new Ord( new F>() { public F f(final Boolean a1) { return new F() { public Ordering f(final Boolean a2) { final int x = a1.compareTo(a2); return x < 0 ? Ordering.LT : x == 0 ? Ordering.EQ : Ordering.GT; } }; } }); /** * An order instance for the byte type. */ public static final Ord byteOrd = new Ord( new F>() { public F f(final Byte a1) { return new F() { public Ordering f(final Byte a2) { final int x = a1.compareTo(a2); return x < 0 ? Ordering.LT : x == 0 ? Ordering.EQ : Ordering.GT; } }; } }); /** * An order instance for the char type. */ public static final Ord charOrd = new Ord( new F>() { public F f(final Character a1) { return new F() { public Ordering f(final Character a2) { final int x = a1.compareTo(a2); return x < 0 ? Ordering.LT : x == 0 ? Ordering.EQ : Ordering.GT; } }; } }); /** * An order instance for the double type. */ public static final Ord doubleOrd = new Ord( new F>() { public F f(final Double a1) { return new F() { public Ordering f(final Double a2) { final int x = a1.compareTo(a2); return x < 0 ? Ordering.LT : x == 0 ? Ordering.EQ : Ordering.GT; } }; } }); /** * An order instance for the float type. */ public static final Ord floatOrd = new Ord( new F>() { public F f(final Float a1) { return new F() { public Ordering f(final Float a2) { final int x = a1.compareTo(a2); return x < 0 ? Ordering.LT : x == 0 ? Ordering.EQ : Ordering.GT; } }; } }); /** * An order instance for the int type. */ public static final Ord intOrd = new Ord( new F>() { public F f(final Integer a1) { return new F() { public Ordering f(final Integer a2) { final int x = a1.compareTo(a2); return x < 0 ? Ordering.LT : x == 0 ? Ordering.EQ : Ordering.GT; } }; } }); /** * An order instance for the BigInteger type. */ public static final Ord bigintOrd = new Ord( new F>() { public F f(final BigInteger a1) { return new F() { public Ordering f(final BigInteger a2) { final int x = a1.compareTo(a2); return x < 0 ? Ordering.LT : x == 0 ? Ordering.EQ : Ordering.GT; } }; } }); /** * An order instance for the BigDecimal type. */ public static final Ord bigdecimalOrd = new Ord( new F>() { public F f(final BigDecimal a1) { return new F() { public Ordering f(final BigDecimal a2) { final int x = a1.compareTo(a2); return x < 0 ? Ordering.LT : x == 0 ? Ordering.EQ : Ordering.GT; } }; } }); /** * An order instance for the long type. */ public static final Ord longOrd = new Ord( new F>() { public F f(final Long a1) { return new F() { public Ordering f(final Long a2) { final int x = a1.compareTo(a2); return x < 0 ? Ordering.LT : x == 0 ? Ordering.EQ : Ordering.GT; } }; } }); /** * An order instance for the short type. */ public static final Ord shortOrd = new Ord( new F>() { public F f(final Short a1) { return new F() { public Ordering f(final Short a2) { final int x = a1.compareTo(a2); return x < 0 ? Ordering.LT : x == 0 ? Ordering.EQ : Ordering.GT; } }; } }); /** * An order instance for the {@link Ordering} type. */ public static final Ord orderingOrd = new Ord(curry(new F2() { public Ordering f(final Ordering o1, final Ordering o2) { return o1 == o2 ? Ordering.EQ : o1 == Ordering.LT ? Ordering.LT : o2 == Ordering.LT ? Ordering.GT : o1 == Ordering.EQ ? Ordering.LT : Ordering.GT; } })); /** * An order instance for the {@link String} type. */ public static final Ord stringOrd = new Ord( new F>() { public F f(final String a1) { return new F() { public Ordering f(final String a2) { final int x = a1.compareTo(a2); return x < 0 ? Ordering.LT : x == 0 ? Ordering.EQ : Ordering.GT; } }; } }); /** * An order instance for the {@link StringBuffer} type. */ public static final Ord stringBufferOrd = new Ord(new F>() { public F f(final StringBuffer a1) { return new F() { public Ordering f(final StringBuffer a2) { return stringOrd.compare(a1.toString(), a2.toString()); } }; } }); /** * An order instance for the {@link StringBuffer} type. */ public static final Ord stringBuilderOrd = new Ord(new F>() { public F f(final StringBuilder a1) { return new F() { public Ordering f(final StringBuilder a2) { return stringOrd.compare(a1.toString(), a2.toString()); } }; } }); /** * An order instance for the {@link Option} type. * * @param oa Order across the element of the option. * @return An order instance for the {@link Option} type. */ public static Ord> optionOrd(final Ord oa) { return new Ord>(new F, F, Ordering>>() { public F, Ordering> f(final Option o1) { return new F, Ordering>() { public Ordering f(final Option o2) { return o1.isNone() ? o2.isNone() ? Ordering.EQ : Ordering.LT : o2.isNone() ? Ordering.GT : oa.f.f(o1.some()).f(o2.some()); } }; } }); } /** * An order instance for the {@link Either} type. * * @param oa Order across the left side of {@link Either}. * @param ob Order across the right side of {@link Either}. * @return An order instance for the {@link Either} type. */ public static Ord> eitherOrd(final Ord oa, final Ord ob) { return new Ord>(new F, F, Ordering>>() { public F, Ordering> f(final Either e1) { return new F, Ordering>() { public Ordering f(final Either e2) { return e1.isLeft() ? e2.isLeft() ? oa.f.f(e1.left().value()).f(e2.left().value()) : Ordering.LT : e2.isLeft() ? Ordering.GT : ob.f.f(e1.right().value()).f(e2.right().value()); } }; } }); } /** * An order instance for the {@link Validation} type. * * @param oa Order across the failing side of {@link Validation}. * @param ob Order across the succeeding side of {@link Validation}. * @return An order instance for the {@link Validation} type. */ public static Ord> validationOrd(final Ord oa, final Ord ob) { return eitherOrd(oa, ob).comap(Validation.either()); } /** * An order instance for the {@link List} type. * * @param oa Order across the elements of the list. * @return An order instance for the {@link List} type. */ public static Ord> listOrd(final Ord oa) { return new Ord>(new F, F, Ordering>>() { public F, Ordering> f(final List l1) { return new F, Ordering>() { public Ordering f(final List l2) { if (l1.isEmpty()) return l2.isEmpty() ? Ordering.EQ : Ordering.LT; else if (l2.isEmpty()) return l1.isEmpty() ? Ordering.EQ : Ordering.GT; else { final Ordering c = oa.compare(l1.head(), l2.head()); return c == Ordering.EQ ? listOrd(oa).f.f(l1.tail()).f(l2.tail()) : c; } } }; } }); } /** * An order instance for the {@link NonEmptyList} type. * * @param oa Order across the elements of the non-empty list. * @return An order instance for the {@link NonEmptyList} type. */ public static Ord> nonEmptyListOrd(final Ord oa) { return listOrd(oa).comap(NonEmptyList.toList_()); } /** * An order instance for the {@link Stream} type. * * @param oa Order across the elements of the stream. * @return An order instance for the {@link Stream} type. */ public static Ord> streamOrd(final Ord oa) { return new Ord>(new F, F, Ordering>>() { public F, Ordering> f(final Stream s1) { return new F, Ordering>() { public Ordering f(final Stream s2) { if (s1.isEmpty()) return s2.isEmpty() ? Ordering.EQ : Ordering.LT; else if (s2.isEmpty()) return s1.isEmpty() ? Ordering.EQ : Ordering.GT; else { final Ordering c = oa.compare(s1.head(), s2.head()); return c == Ordering.EQ ? streamOrd(oa).f.f(s1.tail()._1()).f(s2.tail()._1()) : c; } } }; } }); } /** * An order instance for the {@link Array} type. * * @param oa Order across the elements of the array. * @return An order instance for the {@link Array} type. */ public static Ord> arrayOrd(final Ord oa) { return new Ord>(new F, F, Ordering>>() { public F, Ordering> f(final Array a1) { return new F, Ordering>() { public Ordering f(final Array a2) { int i = 0; //noinspection ForLoopWithMissingComponent for (; i < a1.length() && i < a2.length(); i++) { final Ordering c = oa.compare(a1.get(i), a2.get(i)); if (c == Ordering.GT || c == Ordering.LT) return c; } return i == a1.length() ? i == a2.length() ? Ordering.EQ : Ordering.LT : i == a1.length() ? Ordering.EQ : Ordering.GT; } }; } }); } /** * An order instance for the {@link Set} type. * * @param oa Order across the elements of the set. * @return An order instance for the {@link Set} type. */ public static Ord> setOrd(final Ord oa) { return streamOrd(oa).comap(new F, Stream>() { public Stream f(final Set as) { return as.toStream(); } }); } /** * An order instance for the {@link Unit} type. */ public static final Ord unitOrd = ord(curry(new F2() { public Ordering f(final Unit u1, final Unit u2) { return Ordering.EQ; } })); /** * An order instance for a product-1. * * @param oa Order across the produced type. * @return An order instance for a product-1. */ public static Ord> p1Ord(final Ord oa) { return oa.comap(P1.__1()); } /** * An order instance for a product-2, with the first factor considered most significant. * * @param oa An order instance for the first factor. * @param ob An order instance for the second factor. * @return An order instance for a product-2, with the first factor considered most significant. */ public static Ord> p2Ord(final Ord oa, final Ord ob) { return ord(curry(new F2, P2, Ordering>() { public Ordering f(final P2 a, final P2 b) { return oa.eq(a._1(), b._1()) ? ob.compare(a._2(), b._2()) : oa.compare(a._1(), b._1()); } })); } /** * An order instance for a product-3, with the first factor considered most significant. * * @param oa An order instance for the first factor. * @param ob An order instance for the second factor. * @param oc An order instance for the third factor. * @return An order instance for a product-3, with the first factor considered most significant. */ public static Ord> p3Ord(final Ord oa, final Ord ob, final Ord oc) { return ord(curry(new F2, P3, Ordering>() { public Ordering f(final P3 a, final P3 b) { return oa.eq(a._1(), b._1()) ? p2Ord(ob, oc).compare(P.p(a._2(), a._3()), P.p(b._2(), b._3())) : oa.compare(a._1(), b._1()); } })); } /** * An order instance for the Natural type. */ public static final Ord naturalOrd = bigintOrd.comap(Natural.bigIntegerValue); /** * An order instance for the Comparable interface. * * @return An order instance for the Comparable interface. */ public static > Ord comparableOrd() { return ord(new F>() { public F f(final A a1) { return new F() { public Ordering f(final A a2) { final int x = a1.compareTo(a2); return x < 0 ? Ordering.LT : x == 0 ? Ordering.EQ : Ordering.GT; } }; } }); } /** * An order instance that uses {@link Object#hashCode()} for computing the order and equality, * thus objects returning the same hashCode are considered to be equals (check {@link #hashEqualsOrd()} * for an additional check on {@link Object#equals(Object)}). * * @return An order instance that is based on {@link Object#hashCode()}. * @see #hashEqualsOrd() */ public static Ord hashOrd() { return Ord. ord(new F>() { @Override public F f(final A a) { return new F() { @Override public Ordering f(final A a2) { final int x = a.hashCode() - a2.hashCode(); return x < 0 ? Ordering.LT : x == 0 ? Ordering.EQ : Ordering.GT; } }; } }); } /** * An order instance that uses {@link Object#hashCode()} and {@link Object#equals()} for computing * the order and equality. First the hashCode is compared, if this is equal, objects are compared * using {@link Object#equals()}. * * @return An order instance that is based on {@link Object#hashCode()} and {@link Object#equals()}. */ public static Ord hashEqualsOrd() { return Ord. ord(new F>() { @Override public F f(final A a) { return new F() { @Override public Ordering f(final A a2) { final int x = a.hashCode() - a2.hashCode(); return x < 0 ? Ordering.LT : x == 0 && a.equals(a2) ? Ordering.EQ : Ordering.GT; } }; } }); } }