我正在使用URLClassLoader来加载类。
虽然我传递的文件的名称已成功加载,但是对于它所引用的类,会抛出NoClassDefFoundError。这两个类都是不同jar文件的一部分。
更奇怪的是,同一段代码在Windows上工作正常,但它在Linux上抛出错误。
以下是我正在使用的代码:
package com.pb.spectrum.lim.upgrade.common;
import java.io.File;
import java.io.FilenameFilter;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.net.MalformedURLException;
import java.net.URL;
import java.net.URLClassLoader;
import java.util.ArrayList;
import java.util.List;
public class ClassInvoker {
public static void main(String[] args) throws MalformedURLException, ClassNotFoundException, NoSuchMethodException, SecurityException, IllegalAccessException, IllegalArgumentException, InvocationTargetException {
if(args.length < 2){
throw new IllegalArgumentException("First argument should be Class name. Second argument should be jar's directory. Rest of the arguments passed to main method of invoked class.");
}
File jarsDir = new File(args[1]);
if(!jarsDir.isDirectory()){
throw new IllegalArgumentException("Second argument should point to jar's directory.");
}
URLClassLoader urlClassLoader = null;
List<URL> urls = new ArrayList<URL>();
File[] files = jarsDir.listFiles(new FilenameFilter() {
@Override
public boolean accept(File dir, String name) {
return name.toLowerCase().endsWith(".jar");
}
});
for(File jarFile : files){
urls.add(jarFile.toURI().toURL());
}
urlClassLoader = new URLClassLoader(urls.toArray(new URL[urls.size()]));
Thread.currentThread().setContextClassLoader(urlClassLoader);
Object[] params = new String[args.length-2];
// First 2 arguments are not required for class to be invoked.
System.arraycopy(args, 2, params, 0, params.length);
for(Object string : params){
System.out.println("Param ---->" + (String)string);
}
Class clazz = urlClassLoader.loadClass(args[0]);
Method method = clazz.getMethod("main", new Class[]{String[].class});
method.invoke(null, new Object[] {params});
}
}
现在如果我传递类作为A.B.MyClass1加载,它会成功加载,但无法加载A.B.MyClass2,它位于单独的jar文件中。
答案 0 :(得分:0)
我的问题已经解决,但我不确定这个问题的原因。
总结:
正在加载的罐子序列导致问题。在我的特殊情况下,在将列表提供给URLClassLoader之前对jar的顺序进行排序可以解决问题。
但它提出了另一个问题
以下是详细信息
我有两个具有相同包结构的jar文件,如A.B但只有一个具有MyClass2。现在问题的原因是jar没有在具有MyClass2的jar之前加载MyClass2并且以某种方式(我不知道如何或为什么)URLClassLoader看到第一个jar有包A.B但没有MyClass2类并抛出NoClassDefFoundError。在排序之后,首先加载具有MyClass2的jar,然后使用ClassLoader来定位该类。
我不确定这种行为。我自己也很困惑。