![JAR search and dependency download from the Maven repository](/logo.png)
bboss.org.apache.velocity.util.introspection.IntrospectionUtils Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of bboss-velocity Show documentation
Show all versions of bboss-velocity Show documentation
bboss is a j2ee framework include aop/ioc,mvc,persistent,taglib,rpc,event ,bean-xml serializable and so on.http://www.bbossgroups.com
The newest version!
package bboss.org.apache.velocity.util.introspection;
/*
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you 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.
*/
import org.apache.commons.lang3.reflect.TypeUtils;
import java.lang.reflect.Array;
import java.lang.reflect.GenericArrayType;
import java.lang.reflect.ParameterizedType;
import java.lang.reflect.Type;
import java.lang.reflect.TypeVariable;
import java.lang.reflect.WildcardType;
import java.util.HashMap;
import java.util.Map;
/**
*
* @author Jason van Zyl
* @author Bob McWhirter
* @author Christoph Reck
* @author Geir Magnusson Jr.
* @author Attila Szegedi
* @author Nathan Bubna
* @author Claude Brisson
* @version $Id: IntrospectionUtils.java 476785 2006-11-19 10:06:21Z henning $
* @since 1.6
*/
public class IntrospectionUtils
{
/**
* boxing helper maps for standard types
*/
static Map, Class>> boxingMap, unboxingMap;
static
{
boxingMap = new HashMap<>();
boxingMap.put(Boolean.TYPE, Boolean.class);
boxingMap.put(Character.TYPE, Character.class);
boxingMap.put(Byte.TYPE, Byte.class);
boxingMap.put(Short.TYPE, Short.class);
boxingMap.put(Integer.TYPE, Integer.class);
boxingMap.put(Long.TYPE, Long.class);
boxingMap.put(Float.TYPE, Float.class);
boxingMap.put(Double.TYPE, Double.class);
unboxingMap = new HashMap<>();
for (Map.Entry, Class>> entry : boxingMap.entrySet())
{
unboxingMap.put(entry.getValue(), entry.getKey());
}
}
/**
* returns boxed type (or input type if not a primitive type)
* @param clazz input class
* @return boxed class
*/
public static Class> getBoxedClass(Class clazz)
{
Class> boxed = boxingMap.get(clazz);
return boxed == null ? clazz : boxed;
}
/**
* returns unboxed type (or input type if not successful)
* @param clazz input class
* @return unboxed class
*/
public static Class> getUnboxedClass(Class clazz)
{
Class> unboxed = unboxingMap.get(clazz);
return unboxed == null ? clazz : unboxed;
}
/**
* returns the Class corresponding to a Type, if possible
* @param type the input Type
* @return found Class, if any
*/
public static Class> getTypeClass(Type type)
{
if (type == null)
{
return null;
}
if (type instanceof Class>)
{
return (Class>)type;
}
else if (type instanceof ParameterizedType)
{
return (Class>)((ParameterizedType)type).getRawType();
}
else if (type instanceof GenericArrayType)
{
Type componentType = ((GenericArrayType)type).getGenericComponentType();
Class> componentClass = getTypeClass(componentType);
if (componentClass != null)
{
return Array.newInstance(componentClass, 0).getClass();
}
}
else if (type instanceof TypeVariable)
{
Type[] bounds = TypeUtils.getImplicitBounds((TypeVariable)type);
if (bounds.length == 1) return getTypeClass(bounds[0]);
}
else if (type instanceof WildcardType)
{
Type[] bounds = TypeUtils.getImplicitUpperBounds((WildcardType)type);
if (bounds.length == 1) return getTypeClass(bounds[0]);
}
return null;
}
/**
* Determines whether a type represented by a class object is
* convertible to another type represented by a class object using a
* method invocation conversion, treating object types of primitive
* types as if they were primitive types (that is, a Boolean actual
* parameter type matches boolean primitive formal type). This behavior
* is because this method is used to determine applicable methods for
* an actual parameter list, and primitive types are represented by
* their object duals in reflective method calls.
*
* @param formal the formal parameter type to which the actual
* parameter type should be convertible
* @param actual the actual parameter type.
* @param possibleVarArg whether or not we're dealing with the last parameter
* in the method declaration
* @return true if either formal type is assignable from actual type,
* or formal is a primitive type and actual is its corresponding object
* type or an object type of a primitive type that can be converted to
* the formal type.
*/
public static boolean isMethodInvocationConvertible(Type formal,
Class> actual,
boolean possibleVarArg)
{
Class> formalClass = getTypeClass(formal);
if (formalClass != null)
{
/* if it's a null, it means the arg was null */
if (actual == null)
{
return !formalClass.isPrimitive();
}
/* Check for identity or widening reference conversion */
if (formalClass.isAssignableFrom(actual))
{
return true;
}
/* 2.0: Since MethodMap's comparison functions now use this method with potentially reversed arguments order,
* actual can be a primitive type. */
/* Check for boxing */
if (!formalClass.isPrimitive() && actual.isPrimitive())
{
Class> boxed = boxingMap.get(actual);
if (boxed != null && (boxed == formalClass || formalClass.isAssignableFrom(boxed))) return true;
}
if (formalClass.isPrimitive())
{
if (actual.isPrimitive())
{
/* check for widening primitive conversion */
if (formalClass == Short.TYPE && actual == Byte.TYPE)
return true;
if (formalClass == Integer.TYPE && (
actual == Byte.TYPE || actual == Short.TYPE))
return true;
if (formalClass == Long.TYPE && (
actual == Byte.TYPE || actual == Short.TYPE || actual == Integer.TYPE))
return true;
if (formalClass == Float.TYPE && (
actual == Byte.TYPE || actual == Short.TYPE || actual == Integer.TYPE ||
actual == Long.TYPE))
return true;
if (formalClass == Double.TYPE && (
actual == Byte.TYPE || actual == Short.TYPE || actual == Integer.TYPE ||
actual == Long.TYPE || actual == Float.TYPE))
return true;
} else
{
/* Check for unboxing with widening primitive conversion. */
if (formalClass == Boolean.TYPE && actual == Boolean.class)
return true;
if (formalClass == Character.TYPE && actual == Character.class)
return true;
if (formalClass == Byte.TYPE && actual == Byte.class)
return true;
if (formalClass == Short.TYPE && (actual == Short.class || actual == Byte.class))
return true;
if (formalClass == Integer.TYPE && (actual == Integer.class || actual == Short.class ||
actual == Byte.class))
return true;
if (formalClass == Long.TYPE && (actual == Long.class || actual == Integer.class ||
actual == Short.class || actual == Byte.class))
return true;
if (formalClass == Float.TYPE && (actual == Float.class || actual == Long.class ||
actual == Integer.class || actual == Short.class || actual == Byte.class))
return true;
if (formalClass == Double.TYPE && (actual == Double.class || actual == Float.class ||
actual == Long.class || actual == Integer.class || actual == Short.class ||
actual == Byte.class))
return true;
}
}
/* Check for vararg conversion. */
if (possibleVarArg && formalClass.isArray())
{
if (actual.isArray())
{
actual = actual.getComponentType();
}
return isMethodInvocationConvertible(formalClass.getComponentType(),
actual, false);
}
return false;
}
else
{
// no distinction between strict and implicit, not a big deal in this case
if (TypeUtils.isAssignable(actual, formal))
{
return true;
}
return possibleVarArg && TypeUtils.isArrayType(formal) &&
TypeUtils.isAssignable(actual, TypeUtils.getArrayComponentType(formal));
}
}
/**
* Determines whether a type represented by a class object is
* convertible to another type represented by a class object using a
* method invocation conversion, without matching object and primitive
* types. This method is used to determine the more specific type when
* comparing signatures of methods.
*
* @param formal the formal parameter type to which the actual
* parameter type should be convertible
* @param actual the actual parameter type.
* @param possibleVarArg whether or not we're dealing with the last parameter
* in the method declaration
* @return true if either formal type is assignable from actual type,
* or formal and actual are both primitive types and actual can be
* subject to widening conversion to formal.
*/
public static boolean isStrictMethodInvocationConvertible(Type formal,
Class> actual,
boolean possibleVarArg)
{
Class> formalClass = getTypeClass(formal);
if (formalClass != null)
{
/* Check for nullity */
if (actual == null)
{
return !formalClass.isPrimitive();
}
/* Check for identity or widening reference conversion */
if (formalClass.isAssignableFrom(actual))
{
return true;
}
/* Check for widening primitive conversion. */
if (formalClass.isPrimitive())
{
if (formal == Short.TYPE && (actual == Byte.TYPE))
return true;
if (formal == Integer.TYPE &&
(actual == Short.TYPE || actual == Byte.TYPE))
return true;
if (formal == Long.TYPE &&
(actual == Integer.TYPE || actual == Short.TYPE ||
actual == Byte.TYPE))
return true;
if (formal == Float.TYPE &&
(actual == Long.TYPE || actual == Integer.TYPE ||
actual == Short.TYPE || actual == Byte.TYPE))
return true;
if (formal == Double.TYPE &&
(actual == Float.TYPE || actual == Long.TYPE ||
actual == Integer.TYPE || actual == Short.TYPE ||
actual == Byte.TYPE))
return true;
}
/* Check for vararg conversion. */
if (possibleVarArg && formalClass.isArray())
{
if (actual.isArray())
{
actual = actual.getComponentType();
}
return isStrictMethodInvocationConvertible(formalClass.getComponentType(),
actual, false);
}
return false;
}
else
{
// no distinction between strict and implicit, not a big deal in this case
if (TypeUtils.isAssignable(actual, formal))
{
return true;
}
return possibleVarArg && TypeUtils.isArrayType(formal) &&
TypeUtils.isAssignable(actual, TypeUtils.getArrayComponentType(formal));
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy