我正在开发一个项目,该项目需要从文件系统动态加载类,我用Google搜索并发现我需要使用自定义ClassLoader
。我已经实现了自己的类加载器,当我在控制台上运行它时工作正常,问题是当我尝试在服务器上部署应用程序时,它导致ClassNotFoundException
。
问题是我尝试加载的类包含对应用程序已经加载的另一个类的一些引用,但是它试图从同一位置加载引用。
public class HandlerClassLoader extends ClassLoader {
static Logger log = Logger.getLogger(HandlerClassLoader.class.getName());
URLClassLoader ucl;
private ProcessDefinition processDefinition = null;
boolean flag = false;
public URL[] loadJars() throws MalformedURLException {
PropertiesFiles propFiles = new PropertiesFiles();
File f = new File(propFiles.getValue("jarslocation"));
log.debug("Loading JARS files from " + f.getAbsolutePath());
List<URL> urls = new ArrayList<URL>();
String filesName[] = f.list();
for (String jars : filesName)
if (jars.endsWith("jar")) {
urls.add(new URL("file:///"
+ propFiles.getValue("jarslocation") + "/" + jars));
}
URL[] array = urls.toArray(new URL[urls.size()]);
return array;
}
public HandlerClassLoader() {
super(Thread.currentThread().getContextClassLoader());
log.debug("Called to the " + this.getClass().getName()
+ " No Parameter Constructor");
try {
ucl = new URLClassLoader(loadJars());
} catch (MalformedURLException e) {
e.printStackTrace();
}
}
public HandlerClassLoader(ClassLoader parent,
ProcessDefinition processDefinition) {
super(parent);
log.debug("Called to the " + this.getClass().getName()
+ " Parameterized Constructor");
try {
ucl = new URLClassLoader(loadJars());
} catch (MalformedURLException e) {
e.printStackTrace();
}
this.processDefinition = processDefinition;
}
public Class<?> findClass(String className) throws ClassNotFoundException {
log.debug("findClass method of " + this.getClass().getName()
+ " is called with className : " + className);
return ucl.loadClass(className);
}
我认为委托原则不起作用,或者服务器必须具有不同的类加载器实现。
答案 0 :(得分:1)
最有可能的是,您没有委托给正确的父母。我认为选择Thread.currentThread().getContextClassLoader()
会导致在应用程序服务器上使用错误的类加载器,而这只是从控制台应用程序运行时的系统类加载器(类路径)。
因此,您需要确保您提交给自定义类加载器的父级能够看到您打算通过它加载的类。
最后,实现自己的类加载器是一项棘手的工作。例如,您没有考虑定位资源或定义包。您正在使用的第三方库可能都需要这两个库。如果您确实只需要从磁盘加载文件,请考虑使用URLClassLoader
。