com.mchange.v1.lang.ClassUtils Maven / Gradle / Ivy
/*
* Distributed as part of mchange-commons-java 0.2.11
*
* Copyright (C) 2015 Machinery For Change, Inc.
*
* Author: Steve Waldman
*
* This library is free software; you can redistribute it and/or modify
* it under the terms of EITHER:
*
* 1) The GNU Lesser General Public License (LGPL), version 2.1, as
* published by the Free Software Foundation
*
* OR
*
* 2) The Eclipse Public License (EPL), version 1.0
*
* You may choose which license to accept if you wish to redistribute
* or modify this work. You may offer derivatives of this work
* under the license you have chosen, or you may provide the same
* choice of license which you have been offered here.
*
* This software is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
*
* You should have received copies of both LGPL v2.1 and EPL v1.0
* along with this software; see the files LICENSE-EPL and LICENSE-LGPL.
* If not, the text of these licenses are currently available at
*
* LGPL v2.1: http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html
* EPL v1.0: http://www.eclipse.org/org/documents/epl-v10.php
*
*/
package com.mchange.v1.lang;
import java.util.*;
import com.mchange.v1.jvm.*;
import java.lang.reflect.Method;
import java.lang.reflect.Modifier;
/**
*
* See also com.mchange.v2.codegen.ClassUtils for related methods.
*
*/
public final class ClassUtils
{
final static String[] EMPTY_SA = new String[0];
static Map primitivesToClasses;
static
{
HashMap tmp = new HashMap();
tmp.put( "boolean", boolean.class );
tmp.put( "int", int.class );
tmp.put( "char", char.class );
tmp.put( "short", short.class );
tmp.put( "int", int.class );
tmp.put( "long", long.class );
tmp.put( "float", float.class );
tmp.put( "double", double.class );
tmp.put( "void", void.class );
primitivesToClasses = Collections.unmodifiableMap( tmp );
}
public static Set publicSupertypesForMethods(Class cl, Method[] methods)
{
Set testClasses = allAssignableFrom( cl );
Set out = new HashSet();
for (Iterator ii = testClasses.iterator(); ii.hasNext(); )
{
Class check = (Class) ii.next();
if ( isPublic( check ) && hasAllMethodsAsSupertype( check, methods ) )
out.add(check);
}
return Collections.unmodifiableSet( out );
}
public static boolean isPublic( Class cl )
{ return ( (cl.getModifiers() & Modifier.PUBLIC) != 0 ); }
public static boolean hasAllMethodsAsSupertype(Class cl, Method[] methods)
{ return hasAllMethods( cl, methods, true ); }
public static boolean hasAllMethodsAsSubtype(Class cl, Method[] methods)
{ return hasAllMethods( cl, methods, false ); }
private static boolean hasAllMethods(Class cl, Method[] methods, boolean cl_as_supertype)
{
for (int i = 0, len = methods.length; i < len; ++i)
if ( !containsMethod( cl, methods[i], cl_as_supertype) )
return false;
return true;
}
public static boolean containsMethodAsSupertype(Class cl, Method m)
{ return containsMethod( cl, m, true ); }
public static boolean containsMethodAsSubtype(Class cl, Method m)
{ return containsMethod( cl, m, false ); }
private static boolean containsMethod(Class cl, Method m, boolean cl_as_supertype)
{
Method check;
try
{
check = cl.getMethod( m.getName(), m.getParameterTypes() );
Class mRetType = m.getReturnType();
Class clRetType = check.getReturnType();
// we deal with potentially covariant return types
return ( ( mRetType.equals( clRetType ) ) ||
( cl_as_supertype && clRetType.isAssignableFrom( mRetType ) ) ||
( !cl_as_supertype && mRetType.isAssignableFrom( clRetType ) ) );
}
catch (NoSuchMethodException e)
{ return false; }
}
public static Set allAssignableFrom(Class type)
{
Set out = new HashSet();
//type itself and superclasses (if any)
for (Class cl = type; cl != null; cl = cl.getSuperclass())
out.add( cl );
//super interfaces (if any)
addSuperInterfacesToSet( type, out );
return out;
}
public static String simpleClassName(Class cl)
{
String scn;
int array_level = 0;
while (cl.isArray())
{
++array_level;
cl = cl.getComponentType();
}
scn = simpleClassName( cl.getName() );
if ( array_level > 0 )
{
StringBuffer sb = new StringBuffer(16);
sb.append( scn );
for( int i = 0; i < array_level; ++i)
sb.append("[]");
return sb.toString();
}
else
return scn;
}
private static String simpleClassName(String fqcn)
{
int pkgdot = fqcn.lastIndexOf('.');
if (pkgdot < 0)
return fqcn;
String scn = fqcn.substring(pkgdot + 1);
if (scn.indexOf('$') >= 0)
{
StringBuffer sb = new StringBuffer(scn);
for (int i = 0, len = sb.length(); i < len; ++i)
{
if (sb.charAt(i) == '$')
sb.setCharAt(i, '.');
}
return sb.toString();
}
else
return scn;
}
public static boolean isPrimitive(String typeStr)
{ return (primitivesToClasses.get( typeStr ) != null); }
public static Class classForPrimitive(String typeStr)
{ return (Class) primitivesToClasses.get( typeStr ); }
public static Class forName(String fqcnOrPrimitive ) throws ClassNotFoundException
{
Class out = classForPrimitive( fqcnOrPrimitive );
if (out == null)
out = Class.forName( fqcnOrPrimitive );
return out;
}
public static Class forName( String fqOrSimple, String[] importPkgs, String[] importClasses )
throws AmbiguousClassNameException, ClassNotFoundException
{
try
{ return Class.forName( fqOrSimple ); }
catch ( ClassNotFoundException e )
{ return classForSimpleName( fqOrSimple, importPkgs, importClasses ); }
}
public static Class classForSimpleName( String simpleName, String[] importPkgs, String[] importClasses )
throws AmbiguousClassNameException, ClassNotFoundException
{
Set checkSet = new HashSet();
Class out = classForPrimitive( simpleName );
if (out == null)
{
if (importPkgs == null)
importPkgs = EMPTY_SA;
if (importClasses == null)
importClasses = EMPTY_SA;
for (int i = 0, len = importClasses.length; i < len; ++i)
{
String importSimpleName = fqcnLastElement( importClasses[i] );
if (! checkSet.add( importSimpleName ) )
throw new IllegalArgumentException("Duplicate imported classes: " +
importSimpleName);
if ( simpleName.equals( importSimpleName ) )
//we won't duplicate assign. we'd have caught it above
out = Class.forName( importClasses[i] );
}
if (out == null)
{
try { out = Class.forName("java.lang." + simpleName); }
catch (ClassNotFoundException e)
{ /* just means we haven't found it yet */ }
for (int i = 0, len = importPkgs.length; i < len; ++i)
{
try
{
String tryClass = importPkgs[i] + '.' + simpleName;
Class test = Class.forName( tryClass );
if ( out == null )
out = test;
else
throw new AmbiguousClassNameException( simpleName, out, test );
}
catch (ClassNotFoundException e)
{ /* just means we haven't found it yet */ }
}
}
}
if (out == null)
throw new ClassNotFoundException( "Could not find a class whose unqualified name is \042" +
simpleName + "\042 with the imports supplied. Import packages are " +
Arrays.asList( importPkgs ) + "; class imports are " +
Arrays.asList( importClasses ) );
else
return out;
}
public static String resolvableTypeName( Class type, String[] importPkgs, String[] importClasses )
throws ClassNotFoundException
{
String simpleName = simpleClassName( type );
try
{ classForSimpleName( simpleName, importPkgs, importClasses ); }
catch ( AmbiguousClassNameException e )
{ return type.getName(); }
return simpleName;
}
public static String fqcnLastElement(String fqcn)
{
int pkgdot = fqcn.lastIndexOf('.');
if (pkgdot < 0)
return fqcn;
return fqcn.substring(pkgdot + 1);
}
/* does not add type itself, only its superinterfaces */
private static void addSuperInterfacesToSet(Class type, Set set)
{
Class[] ifaces = type.getInterfaces();
for (int i = 0, len = ifaces.length; i < len; ++i)
{
set.add( ifaces[i] );
addSuperInterfacesToSet( ifaces[i], set );
}
}
private ClassUtils()
{}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy