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

com.helger.commons.lang.ClassHelper Maven / Gradle / Ivy

/*
 * Copyright (C) 2014-2024 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.lang;

import java.io.InputStream;
import java.lang.reflect.Modifier;
import java.math.BigDecimal;
import java.math.BigInteger;
import java.net.URL;

import javax.annotation.Nonnull;
import javax.annotation.Nullable;
import javax.annotation.concurrent.Immutable;

import com.helger.commons.ValueEnforcer;
import com.helger.commons.annotation.Nonempty;
import com.helger.commons.annotation.PresentForCodeCoverage;
import com.helger.commons.annotation.ReturnsMutableCopy;
import com.helger.commons.collection.impl.CommonsWeakHashMap;
import com.helger.commons.collection.impl.ICommonsMap;
import com.helger.commons.collection.impl.ICommonsSet;
import com.helger.commons.io.stream.StreamHelper;
import com.helger.commons.string.StringHelper;

/**
 * {@link Class} helper methods.
 *
 * @author Philip Helger
 */
@Immutable
public final class ClassHelper
{
  // WeakHashMap because class is used as a key
  private static final ICommonsMap , Class > PRIMITIVE_TO_WRAPPER = new CommonsWeakHashMap <> (8);
  private static final ICommonsMap , Class > WRAPPER_TO_PRIMITIVE = new CommonsWeakHashMap <> (8);

  static
  {
    _registerPrimitiveMapping (boolean.class, Boolean.class);
    _registerPrimitiveMapping (byte.class, Byte.class);
    _registerPrimitiveMapping (char.class, Character.class);
    _registerPrimitiveMapping (double.class, Double.class);
    _registerPrimitiveMapping (float.class, Float.class);
    _registerPrimitiveMapping (int.class, Integer.class);
    _registerPrimitiveMapping (long.class, Long.class);
    _registerPrimitiveMapping (short.class, Short.class);
  }

  private static void _registerPrimitiveMapping (@Nonnull final Class  aPrimitiveType, @Nonnull final Class  aPrimitiveWrapperType)
  {
    PRIMITIVE_TO_WRAPPER.put (aPrimitiveType, aPrimitiveWrapperType);
    WRAPPER_TO_PRIMITIVE.put (aPrimitiveWrapperType, aPrimitiveType);
  }

  @PresentForCodeCoverage
  private static final ClassHelper INSTANCE = new ClassHelper ();

  private ClassHelper ()
  {}

  public static boolean isPublicClass (@Nullable final Class  aClass)
  {
    if (aClass == null)
      return false;

    // Interfaces or annotations are not allowed
    if (aClass.isInterface () || aClass.isAnnotation ())
      return false;

    // Only public classes are allowed
    if (!isPublic (aClass))
      return false;

    // Abstract classes are not allowed
    if (isAbstractClass (aClass))
      return false;

    return true;
  }

  /**
   * Check if the passed class is public, instancable and has a no-argument
   * constructor.
   *
   * @param aClass
   *        The class to check. May be null.
   * @return true if the class is public, instancable and has a
   *         no-argument constructor that is public.
   */
  public static boolean isInstancableClass (@Nullable final Class  aClass)
  {
    if (!isPublicClass (aClass))
      return false;

    // Check if a default constructor is present
    try
    {
      aClass.getConstructor ((Class  []) null);
    }
    catch (final NoSuchMethodException ex)
    {
      return false;
    }
    return true;
  }

  public static boolean isPublic (@Nullable final Class  aClass)
  {
    return aClass != null && Modifier.isPublic (aClass.getModifiers ());
  }

  /**
   * Check if the passed class is an interface or not. Please note that
   * annotations are also interfaces!
   *
   * @param aClass
   *        The class to check.
   * @return true if the class is an interface (or an annotation)
   */
  public static boolean isInterface (@Nullable final Class  aClass)
  {
    return aClass != null && Modifier.isInterface (aClass.getModifiers ());
  }

  public static boolean isAnnotationClass (@Nullable final Class  aClass)
  {
    return aClass != null && aClass.isAnnotation ();
  }

  public static boolean isEnumClass (@Nullable final Class  aClass)
  {
    return aClass != null && aClass.isEnum ();
  }

  /**
   * Check if the passed class is abstract or not. Note: interfaces and
   * annotations are also considered as abstract whereas arrays are never
   * abstract.
   *
   * @param aClass
   *        The class to check.
   * @return true if the passed class is abstract
   */
  public static boolean isAbstractClass (@Nullable final Class  aClass)
  {
    // Special case for arrays (see documentation of Class.getModifiers: only
    // final and interface are set, the rest is indeterministic)
    return aClass != null && !aClass.isArray () && Modifier.isAbstract (aClass.getModifiers ());
  }

  public static boolean isArrayClass (@Nullable final Class  aClass)
  {
    return aClass != null && aClass.isArray ();
  }

  public static boolean isPrimitiveType (@Nullable final Class  aClass)
  {
    return aClass != null && PRIMITIVE_TO_WRAPPER.containsKey (aClass);
  }

  public static boolean isPrimitiveWrapperType (@Nullable final Class  aClass)
  {
    return aClass != null && WRAPPER_TO_PRIMITIVE.containsKey (aClass);
  }

  /**
   * Get the primitive wrapper class of the passed primitive class.
   *
   * @param aClass
   *        The primitive class. May be null.
   * @return null if the passed class is not a primitive class.
   */
  @Nullable
  public static Class  getPrimitiveWrapperClass (@Nullable final Class  aClass)
  {
    if (isPrimitiveWrapperType (aClass))
      return aClass;
    return PRIMITIVE_TO_WRAPPER.get (aClass);
  }

  /**
   * Get the primitive class of the passed primitive wrapper class.
   *
   * @param aClass
   *        The primitive wrapper class. May be null.
   * @return null if the passed class is not a primitive wrapper
   *         class.
   */
  @Nullable
  public static Class  getPrimitiveClass (@Nullable final Class  aClass)
  {
    if (isPrimitiveType (aClass))
      return aClass;
    return WRAPPER_TO_PRIMITIVE.get (aClass);
  }

  @Nonnull
  @ReturnsMutableCopy
  public static ICommonsSet > getAllPrimitiveClasses ()
  {
    return PRIMITIVE_TO_WRAPPER.copyOfKeySet ();
  }

  @Nonnull
  @ReturnsMutableCopy
  public static ICommonsSet > getAllPrimitiveWrapperClasses ()
  {
    return WRAPPER_TO_PRIMITIVE.copyOfKeySet ();
  }

  public static boolean isStringClass (@Nullable final Class  aClass)
  {
    if (aClass == null)
      return false;
    // Base class of String, StringBuffer and StringBuilder
    return CharSequence.class.isAssignableFrom (aClass);
  }

  public static boolean isCharacterClass (@Nullable final Class  aClass)
  {
    if (aClass == null)
      return false;
    return Character.class.isAssignableFrom (aClass) || char.class.isAssignableFrom (aClass);
  }

  public static boolean isBooleanClass (@Nullable final Class  aClass)
  {
    if (aClass == null)
      return false;
    return Boolean.class.isAssignableFrom (aClass) || boolean.class.isAssignableFrom (aClass);
  }

  public static boolean isFloatingPointClass (@Nullable final Class  aClass)
  {
    if (aClass == null)
      return false;
    return Double.class.isAssignableFrom (aClass) ||
           double.class.isAssignableFrom (aClass) ||
           Float.class.isAssignableFrom (aClass) ||
           float.class.isAssignableFrom (aClass) ||
           BigDecimal.class.isAssignableFrom (aClass);
  }

  public static boolean isIntegerClass (@Nullable final Class  aClass)
  {
    if (aClass == null)
      return false;
    return Byte.class.isAssignableFrom (aClass) ||
           byte.class.isAssignableFrom (aClass) ||
           Integer.class.isAssignableFrom (aClass) ||
           int.class.isAssignableFrom (aClass) ||
           Long.class.isAssignableFrom (aClass) ||
           long.class.isAssignableFrom (aClass) ||
           Short.class.isAssignableFrom (aClass) ||
           short.class.isAssignableFrom (aClass) ||
           BigInteger.class.isAssignableFrom (aClass);
  }

  /**
   * Check if the passed classes are convertible. Includes conversion checks
   * between primitive types and primitive wrapper types.
   *
   * @param aSrcClass
   *        First class. May not be null.
   * @param aDstClass
   *        Second class. May not be null.
   * @return true if the classes are directly convertible.
   */
  public static boolean areConvertibleClasses (@Nonnull final Class  aSrcClass, @Nonnull final Class  aDstClass)
  {
    ValueEnforcer.notNull (aSrcClass, "SrcClass");
    ValueEnforcer.notNull (aDstClass, "DstClass");

    // Same class?
    if (aDstClass.equals (aSrcClass))
      return true;

    // Default assignable
    if (aDstClass.isAssignableFrom (aSrcClass))
      return true;

    // Special handling for "int.class" == "Integer.class" etc.
    if (aDstClass == getPrimitiveWrapperClass (aSrcClass))
      return true;
    if (aDstClass == getPrimitiveClass (aSrcClass))
      return true;

    // Not convertible
    return false;
  }

  /**
   * null-safe helper method to determine the class of an object.
   *
   * @param aObject
   *        The object to query. May be null.
   * @return null if the passed object is null.
   */
  @Nullable
  public static Class  getClass (@Nullable final Object aObject)
  {
    return aObject == null ? null : aObject.getClass ();
  }

  /**
   * null-safe helper method to determine the class name of an
   * object.
   *
   * @param aObject
   *        The object to query. May be null.
   * @return null if the passed object is null.
   */
  @Nullable
  public static String getClassName (@Nullable final Object aObject)
  {
    return aObject == null ? null : aObject.getClass ().getName ();
  }

  /**
   * Get the name of the object's class without the package.
   *
   * @param aObject
   *        The object to get the information from. May be null .
   * @return The local name of the passed object's class.
   */
  @Nullable
  public static String getClassLocalName (@Nullable final Object aObject)
  {
    return aObject == null ? null : getClassLocalName (aObject.getClass ());
  }

  /**
   * Get the name of the class without the package.
   *
   * @param aClass
   *        The class to get the information from. May be null.
   * @return The local name of the passed class.
   */
  @Nullable
  public static String getClassLocalName (@Nullable final Class  aClass)
  {
    return aClass == null ? null : getClassLocalName (aClass.getName ());
  }

  /**
   * Get the name of the class without the package.
   *
   * @param sClassName
   *        The fully qualified name of the class. May be null.
   * @return The local name of the passed class. Never null.
   */
  @Nullable
  public static String getClassLocalName (@Nullable final String sClassName)
  {
    if (sClassName == null)
      return null;

    final int nIndex = sClassName.lastIndexOf ('.');
    return nIndex == -1 ? sClassName : sClassName.substring (nIndex + 1);
  }

  /**
   * Get the name of the package the passed object resides in.
   *
   * @param aObject
   *        The class to get the information from. May be null.
   * @return The package name of the passed object.
   */
  @Nullable
  public static String getClassPackageName (@Nullable final Object aObject)
  {
    return aObject == null ? null : getClassPackageName (aObject.getClass ());
  }

  /**
   * Get the name of the package the passed class resides in.
   *
   * @param aClass
   *        The class to get the information from. May not be null.
   * @return The package name of the passed class.
   */
  @Nullable
  public static String getClassPackageName (@Nullable final Class  aClass)
  {
    return aClass == null ? null : getClassPackageName (aClass.getName ());
  }

  /**
   * Get the name of the package the passed class resides in.
   *
   * @param sClassName
   *        The name class to get the information from. May be null
   *        .
   * @return The package name of the passed class.
   */
  @Nullable
  public static String getClassPackageName (@Nullable final String sClassName)
  {
    if (sClassName == null)
      return null;

    final int nIndex = sClassName.lastIndexOf ('.');
    return nIndex == -1 ? "" : sClassName.substring (0, nIndex);
  }

  /**
   * Get the class name of the passed object. If the object itself is of type
   * {@link Class}, its name is retrieved, other {@link #getClass()} is called.
   *
   * @param aObject
   *        The object who's class name is to be retrieved.
   * @return "null" for a null parameter
   */
  @Nonnull
  @Nonempty
  public static String getSafeClassName (@Nullable final Object aObject)
  {
    if (aObject instanceof Class )
      return ((Class ) aObject).getName ();
    if (aObject != null)
      return aObject.getClass ().getName ();
    return "null";
  }

  /**
   * Convert a package name to a relative directory name.
   *
   * @param aPackage
   *        The package to be converted. May be null.
   * @return The directory name using forward slashes (/) instead of the dots.
   */
  @Nullable
  public static String getDirectoryFromPackage (@Nullable final Package aPackage)
  {
    // No differentiation
    return aPackage == null ? null : getPathFromClass (aPackage.getName ());
  }

  /**
   * Convert a package name to a relative directory name.
   *
   * @param sPackage
   *        The name of the package to be converted. May be null.
   * @return The directory name using forward slashes (/) instead of the dots.
   */
  @Nullable
  public static String getDirectoryFromPackage (@Nullable final String sPackage)
  {
    // No differentiation
    return getPathFromClass (sPackage);
  }

  /**
   * Get the path representation of the passed class. The path representation is
   * achieved by replacing all dots (.) with forward slashes (/) in the class
   * name.
   *
   * @param aClass
   *        The class of which the path is to be retrieved. May be
   *        null.
   * @return The path representation. Never null.
   */
  @Nullable
  public static String getPathFromClass (@Nullable final Class  aClass)
  {
    return aClass == null ? null : getPathFromClass (aClass.getName ());
  }

  /**
   * Get the path representation of the passed class name. The path
   * representation is achieved by replacing all dots (.) with forward slashes
   * (/) in the class name.
   *
   * @param sClassName
   *        The class name of which the path is to be retrieved. May be
   *        null.
   * @return The path representation
   */
  @Nullable
  public static String getPathFromClass (@Nullable final String sClassName)
  {
    return sClassName == null ? null : StringHelper.replaceAll (sClassName, '.', '/');
  }

  /**
   * Get the class name of the passed path. The class name is retrieved by
   * replacing all path separators (\ and /) with dots (.). This method does not
   * handle the file extension, so it's up to the caller to skip of any file
   * extension!
   *
   * @param sPath
   *        The path to be converted. May be null.
   * @return The class name.
   */
  @Nullable
  public static String getClassFromPath (@Nullable final String sPath)
  {
    return sPath == null ? null : StringHelper.replaceMultipleAsString (sPath, new char [] { '\\', '/' }, '.');
  }

  /**
   * Get the hex representation of the passed object's address. Note that this
   * method makes no differentiation between 32 and 64 bit architectures. The
   * result is always a hexadecimal value preceded by "0x" and followed by
   * exactly 8 characters.
   *
   * @param aObject
   *        The object who's address is to be retrieved. May be
   *        null.
   * @return Depending on the current architecture. Always starting with "0x"
   *         and than containing the address.
   * @see System#identityHashCode(Object)
   */
  @Nonnull
  @Nonempty
  public static String getObjectAddress (@Nullable final Object aObject)
  {
    if (aObject == null)
      return "0x00000000";
    return "0x" + StringHelper.getHexStringLeadingZero (System.identityHashCode (aObject), 8);
  }

  @Nonnull
  private static String _getPathWithLeadingSlash (@Nonnull @Nonempty final String sPath)
  {
    return sPath.charAt (0) == '/' ? sPath : "/" + sPath;
  }

  /**
   * Get the URL of the passed resource using the class loader of the specified
   * class only. This is a sanity wrapper around
   * class.getResource (sPath).
   *
   * @param aClass
   *        The class to be used. May not be null.
   * @param sPath
   *        The path to be resolved. May neither be null nor empty.
   *        Internally it is ensured that the provided path does start with a
   *        slash.
   * @return null if the path could not be resolved using the
   *         specified class loader.
   */
  @Nullable
  public static URL getResource (@Nonnull final Class  aClass, @Nonnull @Nonempty final String sPath)
  {
    ValueEnforcer.notNull (aClass, "Class");
    ValueEnforcer.notEmpty (sPath, "Path");

    // Ensure the path does start with a "/"
    final String sPathWithSlash = _getPathWithLeadingSlash (sPath);

    // returns null if not found
    return aClass.getResource (sPathWithSlash);
  }

  /**
   * Get the input stream of the passed resource using the class loader of the
   * specified class only. This is a sanity wrapper around
   * class.getResourceAsStream (sPath).
   *
   * @param aClass
   *        The class to be used. May not be null.
   * @param sPath
   *        The path to be resolved. May neither be null nor empty.
   *        Internally it is ensured that the provided path does start with a
   *        slash.
   * @return null if the path could not be resolved using the
   *         specified class loader.
   */
  @Nullable
  public static InputStream getResourceAsStream (@Nonnull final Class  aClass, @Nonnull @Nonempty final String sPath)
  {
    ValueEnforcer.notNull (aClass, "Class");
    ValueEnforcer.notEmpty (sPath, "Path");

    // Ensure the path does start with a "/"
    final String sPathWithSlash = _getPathWithLeadingSlash (sPath);

    // returns null if not found
    final InputStream aIS = aClass.getResourceAsStream (sPathWithSlash);
    return StreamHelper.checkForInvalidFilterInputStream (aIS);
  }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy