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

com.helger.commons.equals.DefaultEqualsImplementationRegistrarSPI Maven / Gradle / Ivy

/**
 * Copyright (C) 2014-2015 Philip Helger (www.helger.com)
 * philip[at]helger[dot]com
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *         http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */
package com.helger.commons.equals;

import java.io.File;
import java.math.BigDecimal;
import java.math.RoundingMode;
import java.net.URL;
import java.util.Arrays;
import java.util.Collection;
import java.util.Enumeration;
import java.util.Iterator;
import java.util.Locale;
import java.util.Map;
import java.util.concurrent.atomic.AtomicBoolean;
import java.util.concurrent.atomic.AtomicInteger;
import java.util.concurrent.atomic.AtomicLong;

import javax.annotation.Nonnull;

import org.w3c.dom.Node;
import org.w3c.dom.NodeList;

import com.helger.commons.annotation.IsSPIImplementation;
import com.helger.commons.io.file.FilenameHelper;
import com.helger.commons.system.EJavaVersion;

/**
 * This class registers the default equals implementations. The implementations
 * in here should be aligned with the implementations in the
 * {@link com.helger.commons.hashcode.DefaultHashCodeImplementationRegistrarSPI}
 *
 * @author Philip Helger
 */
@IsSPIImplementation
public final class DefaultEqualsImplementationRegistrarSPI implements IEqualsImplementationRegistrarSPI
{
  private static final class EqualsImplementationFile implements IEqualsImplementation
  {
    public boolean areEqual (@Nonnull final Object aObj1, @Nonnull final Object aObj2)
    {
      final File aFile1 = (File) aObj1;
      final File aFile2 = (File) aObj2;
      return FilenameHelper.getCleanPath (aFile1.getAbsoluteFile ())
                           .equals (FilenameHelper.getCleanPath (aFile2.getAbsoluteFile ()));
    }
  }

  private static final class EqualsImplementationEnumeration implements IEqualsImplementation
  {
    public boolean areEqual (final Object aObj1, final Object aObj2)
    {
      final Enumeration  aRealObj1 = (Enumeration ) aObj1;
      final Enumeration  aRealObj2 = (Enumeration ) aObj2;

      while (aRealObj1.hasMoreElements ())
      {
        if (!aRealObj2.hasMoreElements ())
        {
          // Second enumeration is shorter
          return false;
        }
        final Object aChild1 = aRealObj1.nextElement ();
        final Object aChild2 = aRealObj2.nextElement ();
        if (!EqualsImplementationRegistry.areEqual (aChild1, aChild2))
          return false;
      }
      if (aRealObj2.hasMoreElements ())
      {
        // Second enumeration is long
        return false;
      }
      return true;
    }
  }

  private static final class EqualsImplementationIterator implements IEqualsImplementation
  {
    public boolean areEqual (final Object aObj1, final Object aObj2)
    {
      final Iterator  aRealObj1 = (Iterator ) aObj1;
      final Iterator  aRealObj2 = (Iterator ) aObj2;

      while (aRealObj1.hasNext ())
      {
        if (!aRealObj2.hasNext ())
        {
          // Second iterator is shorter
          return false;
        }
        final Object aChild1 = aRealObj1.next ();
        final Object aChild2 = aRealObj2.next ();
        if (!EqualsImplementationRegistry.areEqual (aChild1, aChild2))
          return false;
      }
      if (aRealObj2.hasNext ())
      {
        // Second iterator is longer
        return false;
      }
      return true;
    }
  }

  private static final class EqualsImplementationCollection implements IEqualsImplementation
  {
    public boolean areEqual (final Object aObj1, final Object aObj2)
    {
      final Collection  aRealObj1 = (Collection ) aObj1;
      final Collection  aRealObj2 = (Collection ) aObj2;

      // Size check
      if (aRealObj1.size () != aRealObj2.size ())
        return false;

      // Content check
      final Object [] aData1 = aRealObj1.toArray ();
      final Object [] aData2 = aRealObj2.toArray ();
      return EqualsImplementationRegistry.areEqual (aData1, aData2);
    }
  }

  private static final class EqualsImplementationMap implements IEqualsImplementation
  {
    public boolean areEqual (final Object aObj1, final Object aObj2)
    {
      final Map  aRealObj1 = (Map ) aObj1;
      final Map  aRealObj2 = (Map ) aObj2;

      // Size check
      if (aRealObj1.size () != aRealObj2.size ())
        return false;

      // Content check
      for (final Map.Entry  aEntry1 : aRealObj1.entrySet ())
      {
        final Object aKey1 = aEntry1.getKey ();
        final Object aValue1 = aEntry1.getValue ();
        if (aValue1 == null)
        {
          // Second map must also contain null value
          if (!(aRealObj2.get (aKey1) == null && aRealObj2.containsKey (aKey1)))
            return false;
        }
        else
        {
          // Check value
          final Object aValue2 = aRealObj2.get (aKey1);
          if (!EqualsImplementationRegistry.areEqual (aValue1, aValue2))
            return false;
        }
      }
      return true;
    }
  }

  private static final class EqualsImplementationArrayShort implements IEqualsImplementation
  {
    public boolean areEqual (final Object aObj1, final Object aObj2)
    {
      return Arrays.equals ((short []) aObj1, (short []) aObj2);
    }
  }

  private static final class EqualsImplementationArrayLong implements IEqualsImplementation
  {
    public boolean areEqual (final Object aObj1, final Object aObj2)
    {
      return Arrays.equals ((long []) aObj1, (long []) aObj2);
    }
  }

  private static final class EqualsImplementationArrayInt implements IEqualsImplementation
  {
    public boolean areEqual (final Object aObj1, final Object aObj2)
    {
      return Arrays.equals ((int []) aObj1, (int []) aObj2);
    }
  }

  private static final class EqualsImplementationArrayFloat implements IEqualsImplementation
  {
    public boolean areEqual (final Object aObj1, final Object aObj2)
    {
      return Arrays.equals ((float []) aObj1, (float []) aObj2);
    }
  }

  private static final class EqualsImplementationArrayDouble implements IEqualsImplementation
  {
    public boolean areEqual (final Object aObj1, final Object aObj2)
    {
      return Arrays.equals ((double []) aObj1, (double []) aObj2);
    }
  }

  private static final class EqualsImplementationArrayChar implements IEqualsImplementation
  {
    public boolean areEqual (final Object aObj1, final Object aObj2)
    {
      return Arrays.equals ((char []) aObj1, (char []) aObj2);
    }
  }

  private static final class EqualsImplementationArrayByte implements IEqualsImplementation
  {
    public boolean areEqual (final Object aObj1, final Object aObj2)
    {
      return Arrays.equals ((byte []) aObj1, (byte []) aObj2);
    }
  }

  private static final class EqualsImplementationArrayBoolean implements IEqualsImplementation
  {
    public boolean areEqual (final Object aObj1, final Object aObj2)
    {
      return Arrays.equals ((boolean []) aObj1, (boolean []) aObj2);
    }
  }

  private static final class EqualsImplementationAtomicLong implements IEqualsImplementation
  {
    public boolean areEqual (final Object aObj1, final Object aObj2)
    {
      final AtomicLong aRealObj1 = (AtomicLong) aObj1;
      final AtomicLong aRealObj2 = (AtomicLong) aObj2;
      return aRealObj1.get () == aRealObj2.get ();
    }
  }

  private static final class EqualsImplementationAtomicInteger implements IEqualsImplementation
  {
    public boolean areEqual (final Object aObj1, final Object aObj2)
    {
      final AtomicInteger aRealObj1 = (AtomicInteger) aObj1;
      final AtomicInteger aRealObj2 = (AtomicInteger) aObj2;
      return aRealObj1.get () == aRealObj2.get ();
    }
  }

  private static final class EqualsImplementationAtomicBoolean implements IEqualsImplementation
  {
    public boolean areEqual (final Object aObj1, final Object aObj2)
    {
      final AtomicBoolean aRealObj1 = (AtomicBoolean) aObj1;
      final AtomicBoolean aRealObj2 = (AtomicBoolean) aObj2;
      return aRealObj1.get () == aRealObj2.get ();
    }
  }

  private static final class EqualsImplementationURL implements IEqualsImplementation
  {
    public boolean areEqual (final Object aObj1, final Object aObj2)
    {
      final URL aRealObj1 = (URL) aObj1;
      final URL aRealObj2 = (URL) aObj2;
      return aRealObj1.toExternalForm ().equals (aRealObj2.toExternalForm ());
    }
  }

  private static final class EqualsImplementationNode implements IEqualsImplementation
  {
    public boolean areEqual (final Object aObj1, final Object aObj2)
    {
      final Node aRealObj1 = (Node) aObj1;
      final Node aRealObj2 = (Node) aObj2;
      if (aRealObj1.getNodeType () != aRealObj2.getNodeType ())
        return false;
      if (!EqualsImplementationRegistry.areEqual (aRealObj1.getNodeName (), aRealObj2.getNodeName ()))
        return false;
      if (!EqualsImplementationRegistry.areEqual (aRealObj1.getLocalName (), aRealObj2.getLocalName ()))
        return false;
      if (!EqualsImplementationRegistry.areEqual (aRealObj1.getNamespaceURI (), aRealObj2.getNamespaceURI ()))
        return false;
      if (!EqualsImplementationRegistry.areEqual (aRealObj1.getPrefix (), aRealObj2.getPrefix ()))
        return false;
      if (!EqualsImplementationRegistry.areEqual (aRealObj1.getNodeValue (), aRealObj2.getNodeValue ()))
        return false;

      // For all children
      final NodeList aNL1 = aRealObj1.getChildNodes ();
      final NodeList aNL2 = aRealObj2.getChildNodes ();

      final int nLength = aNL1.getLength ();
      if (nLength != aNL2.getLength ())
        return false;

      for (int i = 0; i < nLength; ++i)
      {
        final Node aChild1 = aNL1.item (i);
        final Node aChild2 = aNL2.item (i);
        if (!EqualsImplementationRegistry.areEqual (aChild1, aChild2))
          return false;
      }

      return true;
    }
  }

  private static final class EqualsImplementationStringBuilder implements IEqualsImplementation
  {
    public boolean areEqual (final Object aObj1, final Object aObj2)
    {
      return aObj1.toString ().equals (aObj2.toString ());
    }
  }

  private static final class EqualsImplementationStringBuffer implements IEqualsImplementation
  {
    public boolean areEqual (final Object aObj1, final Object aObj2)
    {
      return aObj1.toString ().equals (aObj2.toString ());
    }
  }

  private static final class EqualsImplementationFloat implements IEqualsImplementation
  {
    public boolean areEqual (final Object aObj1, final Object aObj2)
    {
      final Float aRealObj1 = (Float) aObj1;
      final Float aRealObj2 = (Float) aObj2;
      return aRealObj1.compareTo (aRealObj2) == 0;
    }
  }

  private static final class EqualsImplementationDouble implements IEqualsImplementation
  {
    public boolean areEqual (final Object aObj1, final Object aObj2)
    {
      final Double aRealObj1 = (Double) aObj1;
      final Double aRealObj2 = (Double) aObj2;
      return aRealObj1.compareTo (aRealObj2) == 0;
    }
  }

  private static final class EqualsImplementationBigDecimal implements IEqualsImplementation
  {
    public boolean areEqual (final Object aObj1, final Object aObj2)
    {
      final BigDecimal aRealObj1 = (BigDecimal) aObj1;
      final BigDecimal aRealObj2 = (BigDecimal) aObj2;
      // Compare is ~15% quicker than the setScale version
      if (true)
        return aRealObj1.compareTo (aRealObj2) == 0;

      final int nMaxScale = Math.max (aRealObj1.scale (), aRealObj2.scale ());
      // Use the same rounding mode for both
      return aRealObj1.setScale (nMaxScale, RoundingMode.HALF_UP).equals (aRealObj2.setScale (nMaxScale,
                                                                                              RoundingMode.HALF_UP));
    }
  }

  private static final class EqualsImplementationLocale implements IEqualsImplementation
  {
    public boolean areEqual (final Object aObj1, final Object aObj2)
    {
      final Locale aRealObj1 = (Locale) aObj1;
      final Locale aRealObj2 = (Locale) aObj2;
      return aRealObj1.toString ().equals (aRealObj2.toString ());
    }
  }

  public void registerEqualsImplementations (@Nonnull final IEqualsImplementationRegistry aRegistry)
  {
    /**
     * Special equals implementation for BigDecimal because
     * BigDecimal.equals returns false if they have a
     * different scale so that "5.5" is not equal "5.50".
     */
    aRegistry.registerEqualsImplementation (BigDecimal.class, new EqualsImplementationBigDecimal ());

    // Special overload for "Double" required!
    aRegistry.registerEqualsImplementation (Double.class, new EqualsImplementationDouble ());

    // Special overload for "Float" required!
    aRegistry.registerEqualsImplementation (Float.class, new EqualsImplementationFloat ());

    // StringBuffer does not implement equals!
    aRegistry.registerEqualsImplementation (StringBuffer.class, new EqualsImplementationStringBuffer ());

    // StringBuilder does not implement equals!
    aRegistry.registerEqualsImplementation (StringBuilder.class, new EqualsImplementationStringBuilder ());

    // Node does not implement equals
    aRegistry.registerEqualsImplementation (Node.class, new EqualsImplementationNode ());

    /**
     * Special equals implementation for URLs because URL.equals
     * performs a host lookup.
* Click here for details */ aRegistry.registerEqualsImplementation (URL.class, new EqualsImplementationURL ()); // AtomicBoolean does not implement equals! aRegistry.registerEqualsImplementation (AtomicBoolean.class, new EqualsImplementationAtomicBoolean ()); // AtomicInteger does not implement equals! aRegistry.registerEqualsImplementation (AtomicInteger.class, new EqualsImplementationAtomicInteger ()); // AtomicLong does not implement equals! aRegistry.registerEqualsImplementation (AtomicLong.class, new EqualsImplementationAtomicLong ()); // Default array implementations // (Object[].class is handled specially!) aRegistry.registerEqualsImplementation (boolean [].class, new EqualsImplementationArrayBoolean ()); aRegistry.registerEqualsImplementation (byte [].class, new EqualsImplementationArrayByte ()); aRegistry.registerEqualsImplementation (char [].class, new EqualsImplementationArrayChar ()); aRegistry.registerEqualsImplementation (double [].class, new EqualsImplementationArrayDouble ()); aRegistry.registerEqualsImplementation (float [].class, new EqualsImplementationArrayFloat ()); aRegistry.registerEqualsImplementation (int [].class, new EqualsImplementationArrayInt ()); aRegistry.registerEqualsImplementation (long [].class, new EqualsImplementationArrayLong ()); aRegistry.registerEqualsImplementation (short [].class, new EqualsImplementationArrayShort ()); // Special handling for Map aRegistry.registerEqualsImplementation (Map.class, new EqualsImplementationMap ()); // Special handling for Collection aRegistry.registerEqualsImplementation (Collection.class, new EqualsImplementationCollection ()); // Special handling for Iterator aRegistry.registerEqualsImplementation (Iterator.class, new EqualsImplementationIterator ()); // Special handling for Enumeration aRegistry.registerEqualsImplementation (Enumeration.class, new EqualsImplementationEnumeration ()); // Special handling for File aRegistry.registerEqualsImplementation (File.class, new EqualsImplementationFile ()); // Special handling for Locale in JDK >= 1.7 if (EJavaVersion.JDK_17.isSupportedVersion ()) aRegistry.registerEqualsImplementation (Locale.class, new EqualsImplementationLocale ()); } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy