URLClassLoader无法在Linux上加载依赖项

时间:2013-03-05 13:34:35

标签: java classloader urlclassloader

我正在使用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文件中。

1 个答案:

答案 0 :(得分:0)

我的问题已经解决,但我不确定这个问题的原因。

总结:

  

正在加载的罐子序列导致问题。在我的特殊情况下,在将列表提供给URLClassLoader之前对jar的顺序进行排序可以解决问题。

但它提出了另一个问题

以下是详细信息

  

我有两个具有相同包结构的jar文件,如A.B但只有一个具有MyClass2。现在问题的原因是jar没有在具有MyClass2的jar之前加载MyClass2并且以某种方式(我不知道如何或为什么)URLClassLoader看到第一个jar有包A.B但没有MyClass2类并抛出NoClassDefFoundError。在排序之后,首先加载具有MyClass2的jar,然后使用ClassLoader来定位该类。

我不确定这种行为。我自己也很困惑。