/*
* Copyright (C) 2006-2013 Bitronix Software (http://www.bitronix.be)
*
* 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 bitronix.tm.utils;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import java.io.InputStream;
import java.util.Arrays;
import java.util.HashSet;
import java.util.Set;
/**
* Static utility methods for loading classes and resources.
*/
public class ClassLoaderUtils {
private final static Logger log = LoggerFactory.getLogger(ClassLoaderUtils.class);
public static Set<Class<?>> getAllInterfaces(Class<?> clazz) {
Set<Class<?>> interfaces = new HashSet<Class<?>>();
for (Class<?> intf : Arrays.asList(clazz.getInterfaces())) {
if (intf.getInterfaces().length > 0) {
interfaces.addAll(getAllInterfaces(intf));
}
interfaces.add(intf);
}
if (clazz.getSuperclass() != null) {
interfaces.addAll(getAllInterfaces(clazz.getSuperclass()));
}
if (clazz.isInterface()) {
interfaces.add(clazz);
}
return interfaces;
}
/**
* Get the class loader which can be used to generate proxies without leaking memory.
* @return the class loader which can be used to generate proxies without leaking memory.
*/
public static ClassLoader getClassLoader() {
ClassLoader cl = Thread.currentThread().getContextClassLoader();
if (cl != null) {
return new CascadingClassLoader(cl);
}
return ClassLoaderUtils.class.getClassLoader();
}
/**
* Load a class by name. Tries the current thread's context loader then falls back to {@link Class#forName(String)}.
* @param className name of the class to load.
* @return the loaded class.
* @throws ClassNotFoundException if the class cannot be found in the classpath.
*/
public static Class loadClass(String className) throws ClassNotFoundException {
ClassLoader cl = Thread.currentThread().getContextClassLoader();
if (cl != null) {
try {
return new CascadingClassLoader(cl).loadClass(className);
} catch (ClassNotFoundException ex) {
if (log.isDebugEnabled()) { log.debug("context classloader could not find class '" + className + "', trying Class.forName() instead"); }
}
}
return Class.forName(className);
}
/**
* Load a resource from the classpath. Tries the current thread's context loader then falls back to
* {@link ClassLoader#getResourceAsStream(String)} using this class' classloader.
* @param resourceName the resource name to load.
* @return a {@link java.io.InputStream} if the resource could be found, null otherwise.
*/
public static InputStream getResourceAsStream(String resourceName) {
ClassLoader cl = Thread.currentThread().getContextClassLoader();
if (cl != null)
return cl.getResourceAsStream(resourceName);
return ClassLoaderUtils.class.getClassLoader().getResourceAsStream(resourceName);
}
private static class CascadingClassLoader extends ClassLoader {
private final ClassLoader contextLoader;
CascadingClassLoader(ClassLoader contextLoader) {
this.contextLoader = contextLoader;
}
@Override
protected Class<?> findClass(String name) throws ClassNotFoundException {
try {
return contextLoader.loadClass(name);
}
catch (ClassNotFoundException cnfe) {
return CascadingClassLoader.class.getClassLoader().loadClass(name);
}
}
}
}