All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.strobel.core.Comparer Maven / Gradle / Ivy

/*
 * Comparer.java
 *
 * Copyright (c) 2012 Mike Strobel
 *
 * This source code is subject to terms and conditions of the Apache License, Version 2.0.
 * A copy of the license can be found in the License.html file at the root of this distribution.
 * By using this source code in any fashion, you are agreeing to be bound by the terms of the
 * Apache License, Version 2.0.
 *
 * You must not remove this notice, or any other, from this software.
 */

package com.strobel.core;

import java.util.Arrays;

/**
 * @author Mike Strobel
 */
public final class Comparer {
    private Comparer() {}

    public static  boolean notEqual(final T o1, final T o2) {
        return o1 == null ? o2 != null
                          : !o1.equals(o2);
    }

    public static  boolean equals(final T o1, final T o2) {
        return o1 == null ? o2 == null
                          : o1.equals(o2);
    }

    public static  boolean referenceEquals(final T o1, final T o2) {
        return o1 == o2;
    }

    public static > int compare(final T o1, final T o2) {
        if (o1 == null) {
            return o2 == null ? 0 : -1;
        }
        return o1.compareTo(o2);
    }

    public static  boolean deepEquals(final T o1, final T o2) {
        if (o1 == o2) {
            return true;
        }

        if (o1 == null || o2 == null) {
            return false;
        }

        return deepEqualsCore(o1, o2);
    }

    public static boolean deepEquals(final Object[] a1, final Object[] a2) {
        if (a1 == a2) {
            return true;
        }

        if (a1 == null || a2 == null) {
            return false;
        }

        final int length = a1.length;

        if (a2.length != length) {
            return false;
        }

        for (int i = 0; i < length; i++) {
            final Object e1 = a1[i];
            final Object e2 = a2[i];

            if (e1 == e2) {
                continue;
            }

            if (e1 == null) {
                return false;
            }

            if (!deepEqualsCore(e1, e2)) {
                return false;
            }
        }

        return true;
    }

    private final static boolean deepEqualsCore(final Object e1, final Object e2) {
        if (e1.getClass().isArray()) {
            if (e1 instanceof Object[] && e2 instanceof Object[]) {
                return deepEquals((Object[]) e1, (Object[]) e2);
            }

            if (e1 instanceof byte[] && e2 instanceof byte[]) {
                return Arrays.equals((byte[]) e1, (byte[]) e2);
            }

            if (e1 instanceof short[] && e2 instanceof short[]) {
                return Arrays.equals((short[]) e1, (short[]) e2);
            }

            if (e1 instanceof int[] && e2 instanceof int[]) {
                return Arrays.equals((int[]) e1, (int[]) e2);
            }

            if (e1 instanceof long[] && e2 instanceof long[]) {
                return Arrays.equals((long[]) e1, (long[]) e2);
            }

            if (e1 instanceof char[] && e2 instanceof char[]) {
                return Arrays.equals((char[]) e1, (char[]) e2);
            }

            if (e1 instanceof float[] && e2 instanceof float[]) {
                return Arrays.equals((float[]) e1, (float[]) e2);
            }

            if (e1 instanceof double[] && e2 instanceof double[]) {
                return Arrays.equals((double[]) e1, (double[]) e2);
            }

            if (e1 instanceof boolean[] && e2 instanceof boolean[]) {
                return Arrays.equals((boolean[]) e1, (boolean[]) e2);
            }
        }
        return e1.equals(e2);
    }

    @SuppressWarnings({"unchecked"})
    public static int compare(final Object a, final Object b) {
        if (a == b) {
            return 0;
        }

        if (a == null) {
            return -1;
        }

        if (b == null) {
            return 1;
        }

        final Class aClass = a.getClass();
        final Class bClass = b.getClass();

        if (Comparable.class.isInstance(a) && aClass.isAssignableFrom(bClass)) {
            return ((Comparable) a).compareTo(b);
        }

        if (Comparable.class.isInstance(b) && bClass.isAssignableFrom(aClass)) {
            return ((Comparable) b).compareTo(a);
        }

        throw new IllegalArgumentException("Values must be comparable.");
    }
}