如何动态地从jar文件运行java类

时间:2015-03-26 13:26:45

标签: java jar classloader

我正在研究一个需要第三方java程序作为服务器运行的java项目。

通常,我会这样做:

java -cp jarfile1.jar:jarfile2.jar className arg1 arg2

然后我运行我的java代码。这样就行了。

我想知道是否有任何方法,包括我的项目所需的两个.jars,直接从我的代码运行该类,而不必手动启动它。

我尝试使用URLClassLoader,正如我在某些示例中看到的那样,但要么我做错了,要么都没有涵盖这个特定的用例。

URLClassLoader classLoader = URLClassLoader.newInstance(new URL[]{new URL("file:///tmp/jarfile1.jar"),new URL("file:///tmp/jarfile2.jar")});
Class<?> cls = classLoader.loadClass("className");
Method method = cls.getDeclaredMethod ("main");
Object instance = cls.newInstance();
Object result = method.invoke (instance);

产量

Exception in thread "main" java.lang.NoClassDefFoundError: alice/tuprolog/lib/InvalidObjectIdException
at java.lang.Class.getDeclaredMethods0(Native Method)
at java.lang.Class.privateGetDeclaredMethods(Class.java:2615)
at java.lang.Class.getDeclaredMethod(Class.java:2007)
at pkg1.MainClass.main(MainClass.java:54)
Caused by: java.lang.ClassNotFoundException: alice.tuprolog.lib.InvalidObjectIdException
    at java.net.URLClassLoader$1.run(URLClassLoader.java:366)
    at java.net.URLClassLoader$1.run(URLClassLoader.java:355)
    at java.security.AccessController.doPrivileged(Native Method)
    at java.net.URLClassLoader.findClass(URLClassLoader.java:354)
    at java.lang.ClassLoader.loadClass(ClassLoader.java:425)
    at sun.misc.Launcher$AppClassLoader.loadClass(Launcher.java:308)
    at java.lang.ClassLoader.loadClass(ClassLoader.java:358)
    ... 4 more

请注意我将.jars复制到/ tmp以隔离失败原因。这些文件存在且可以访问。

如何在java代码中运行上面指定的类?

谢谢!

2 个答案:

答案 0 :(得分:2)

如果该类存在于不同的ClassLoader中,则需要使用反射来获取它:

ClassLoader classLoader = new URLClassLoader(
    new URL[] { firstJarURL, secondJarURL });

String[] args = { arg1, arg2 };

try {
    Class<?> mainClass = classLoader.loadClass("com.somepackage.ClassName");
    mainClass.getMethod("main", String[].class).invoke(null, args);
} catch (ReflectiveOperationException e) {
    throw new RuntimeException(e);
}

答案 1 :(得分:0)

我终于修好了!做的事情:

  1. 我忘了将第二个jar添加到项目的类路径中(duh!)
  2. 由于所有内容都在项目类路径中,我只是重用了当前的类加载器
  3. 最终工作代码:

    ClassLoader classLoader = Thread.currentThread().getContextClassLoader();
    Class<?> cls = classLoader.loadClass("className");
    Method method = cls.getDeclaredMethod("main", String[].class);
    Object instance = cls.newInstance();
    Object result = method.invoke(null, (Object)args);
    

    感谢大家,特别感谢VGR和Joop Eggen在评论中指出第二个jar的错误!

    编辑:正如JB Nizet在评论中指出的那样,直接调用类的main()方法更简单:

    className.main(args);
    

    你已经完成了