J2EE Java

/*
 * Copyright 2005 (C) TJDO.
 * All rights reserved.
 *
 * This software is distributed under the terms of the TJDO License version 1.0.
 * See the terms of the TJDO License in the documentation provided with this software.
 *
 * $Id: Types.java,v 1.1 2005/12/13 20:46:16 jackknifebarber Exp $
 */
import java.math.BigDecimal;
import java.math.BigInteger;
import java.util.HashMap;
/**
 * Contains static methods related to Java and JDO type conversions.
 *
 * @author Mike Martin
 * @version $Revision: 1.1 $
 */
public final class Types
{
    private Types()
    {
    }
    private static final HashMap WRAPPERS_BY_PRIMITIVE = new HashMap(9);
    private static final HashMap PRIMITIVES_BY_WRAPPER = new HashMap(9);
    static
    {
        WRAPPERS_BY_PRIMITIVE.put(Boolean.TYPE,   Boolean.class);
        WRAPPERS_BY_PRIMITIVE.put(Byte.TYPE,      Byte.class);
        WRAPPERS_BY_PRIMITIVE.put(Character.TYPE, Character.class);
        WRAPPERS_BY_PRIMITIVE.put(Short.TYPE,     Short.class);
        WRAPPERS_BY_PRIMITIVE.put(Integer.TYPE,   Integer.class);
        WRAPPERS_BY_PRIMITIVE.put(Long.TYPE,      Long.class);
        WRAPPERS_BY_PRIMITIVE.put(Float.TYPE,     Float.class);
        WRAPPERS_BY_PRIMITIVE.put(Double.TYPE,    Double.class);
        WRAPPERS_BY_PRIMITIVE.put(Void.TYPE,      Void.class);
        PRIMITIVES_BY_WRAPPER.put(Boolean.class,   Boolean.TYPE);
        PRIMITIVES_BY_WRAPPER.put(Byte.class,      Byte.TYPE);
        PRIMITIVES_BY_WRAPPER.put(Character.class, Character.TYPE);
        PRIMITIVES_BY_WRAPPER.put(Short.class,     Short.TYPE);
        PRIMITIVES_BY_WRAPPER.put(Integer.class,   Integer.TYPE);
        PRIMITIVES_BY_WRAPPER.put(Long.class,      Long.TYPE);
        PRIMITIVES_BY_WRAPPER.put(Float.class,     Float.TYPE);
        PRIMITIVES_BY_WRAPPER.put(Double.class,    Double.TYPE);
        PRIMITIVES_BY_WRAPPER.put(Void.class,      Void.TYPE);
    }
    /**
     * Performs a boxing conversion, if applicable.
     * If the type is a Java primitive type it's equivalent wrapper class is
     * returned.
     * Otherwise the type itself is returned.
     *
     * @param c
     *      the type to be boxed, or null
     *
     * @return
     *      The boxed equivalent, or the type itself if not applicable.
     *      null is returned if the argument was null.
     */
    public static Class box(Class c)
    {
        Class boxed = (Class)WRAPPERS_BY_PRIMITIVE.get(c);
        return boxed == null ? c : boxed;
    }
    /**
     * Performs an unboxing conversion, if applicable.
     * If the type is a Java primitive wrapper class it's equivalent primitive
     * type is returned.
     * Otherwise the type itself is returned.
     *
     * @param c
     *      the type to be unboxed, or null
     *
     * @return
     *      The unboxed equivalent, or the type itself if not applicable.
     *      null is returned if the argument was null.
     */
    public static Class unbox(Class c)
    {
        Class unboxed = (Class)PRIMITIVES_BY_WRAPPER.get(c);
        return unboxed == null ? c : unboxed;
    }
    /**
     * Performs a boxing conversion on multiple classes.
     * The returned array contains the results of calling {@link #box(Class)}
     * on every class in the argument array.
     *
     * @param c
     *      the types to be boxed
     *
     * @return
     *      the boxed equivalent of all the types in c
     */
    public static Class[] box(Class[] c)
    {
        int n = c.length;
        Class[] boxed = new Class[n];
        for (int i = 0; i < n; ++i)
            boxed[i] = box(c[i]);
        return boxed;
    }
    /**
     * Performs an unboxing conversion on multiple classes.
     * The returned array contains the results of calling {@link #unbox(Class)}
     * on every class in the argument array.
     *
     * @param c
     *      the types to be unboxed
     *
     * @return
     *      the unboxed equivalent of all the types in c
     */
    public static Class[] unbox(Class[] c)
    {
        int n = c.length;
        Class[] unboxed = new Class[n];
        for (int i = 0; i < n; ++i)
            unboxed[i] = unbox(c[i]);
        return unboxed;
    }
    /**
     * Tests if a set of parameters types are all assignable from a given set of
     * argument types.
     * Returns true iff the number of types match and each parameter type is
     * assignable from its corresponding argument type.
     * "Assignable" is defined by Class.isAssignableFrom().
     * 


     * All classes in the parameter type array and the argument type array must
     * already be boxed.
     *
     * @param boxedParamTypes
     *      the parameter types
     * @param boxedArgTypes
     *      the argument types
     *
     * @return
     *      true if the all the parameter types are assignable from
     *      the argument types.
     */
    public static boolean areAssignableFrom(Class[] boxedParamTypes, Class[] boxedArgTypes)
    {
        int n = boxedParamTypes.length;
        if (n != boxedArgTypes.length)
            return false;
        for (int i = 0; i < n; ++i)
        {
            if (!boxedParamTypes[i].isAssignableFrom(boxedArgTypes[i]))
                return false;
        }
        return true;
    }
    /**
     * Performs unary numeric promotion on the given type.
     * Unary numeric promotion is as defined in §5.6.1 of the Java Language
     * Specification.
     *
     * @param operandType
     *      the type to be promoted, or null
     *
     * @return
     *      The promoted type, or null if the argument was null.
     */
    public static Class unaryNumericPromotion(Class operandType)
    {
        operandType = box(operandType);
        if (operandType == Byte.class || operandType == Character.class || operandType == Short.class)
            return Integer.class;
        else
            return operandType;
    }
    /**
     * Performs binary numeric promotion on a set of input types.
     * Binary numeric promotion is as defined in §14.6.2 of the JDO 2.0
     * spec.
     *
     * @param types
     *      The input types to be promoted.  If more than two types are present
     *      then the first two are promoted, then the promoted type and the
     *      third type are promoted, and so on for all the types.
     *
     * @return
     *      The promoted type.  Returns null if the input array
     *      is empty.  Returns the first type if the input array has only one
     *      type.
     *
     * @exception IllegalArgumentException
     *      if any argument type is not a subclass of java.lang.Number
     */
    public static Class binaryNumericPromotion(Class[] types)
    {
        Class promoted = null;
        int n = types.length;
        if (n > 0)
            promoted = types[0];
        for (int i = 1; i < n; ++i)
            promoted = binaryNumericPromotion(promoted, types[i]);
        return promoted;
    }
    /**
     * Performs binary numeric promotion on a pair of types.
     * Binary numeric promotion is as defined in §14.6.2 of the JDO 2.0
     * spec.
     *
     * @param operand1Type
     *      the first type in the pair.
     * @param operand2Type
     *      the other type in the pair.
     *
     * @return
     *      The promoted type.
     *
     * @exception IllegalArgumentException
     *      if either argument type is not a subclass of java.lang.Number
     */
    public static Class binaryNumericPromotion(Class operand1Type, Class operand2Type)
    {
        operand1Type = box(operand1Type);
        operand2Type = box(operand2Type);
        if (operand1Type == BigDecimal.class || operand2Type == BigDecimal.class)
            return BigDecimal.class;
        if (isBigIntegerVsFloating(operand1Type, operand2Type) || isBigIntegerVsFloating(operand2Type, operand1Type))
            return BigDecimal.class;
        if (operand1Type == BigInteger.class || operand2Type == BigInteger.class)
            return BigInteger.class;
        if (operand1Type == Double.class || operand2Type == Double.class)
            return Double.class;
        if (operand1Type == Float.class || operand2Type == Float.class)
            return Float.class;
        if (operand1Type == Long.class || operand2Type == Long.class)
            return Long.class;
        if (!Number.class.isAssignableFrom(operand1Type) || !Number.class.isAssignableFrom(operand2Type))
            throw new IllegalArgumentException("No defined numeric promotion for operands of type " + operand1Type.getName() + " and " + operand2Type.getName());
        return Integer.class;
    }
    private static boolean isBigIntegerVsFloating(Class c1, Class c2)
    {
        return c1 == BigInteger.class && (c2 == Float.class || c2 == Double.class);
    }
}