【问题标题】:JarClassLoader - loading jars dynamically - how to?JarClassLoader - 动态加载 jars - 如何?
【发布时间】:2015-07-21 15:23:56
【问题描述】:

我已经看到很多类加载器问题,但仍然无法弄清楚为什么,这里的错误。

我正在编写一个使用 2 个版本的 jar 的程序。一个用于从旧存储中获取内容,另一个用于将内容存储在新存储中。

因为我需要一次加载其中一个 jar,所以我使用 JarClassLoader 创建了一个代理来添加一个 jar 并加载它的类。但我面临 ClassNotFoundException。

public class HbaseMigrator  implements Runnable {
   public void run() {
     JarClassLoader jcl = new JarClassLoader();
     jcl.add("hadoop-0.13.0-core-modified-1.jar");
     Object obj1 = JclObjectFactory.getInstance().create(jcl, "UserMigThreadImpl", toProcessQueue,threadName, latch,DBUtil,lock);
     MigThread mig = JclUtils.cast(obj1, MigThread.class, jcl);
     Thread.currentThread().setContextClassLoader(jcl);
     try {
         Method method = MigThread.class.getMethod("callthis", new Class[]{});
         method.invoke(mig, new Object[]{});
        // mig.callthis();
     } catch( Exception e) {
         e.printStackTrace();
     } catch(Error er) {
         er.printStackTrace();
     }
  }
}

调用的方法是:

 public void callthis() {
    DFSUtil = new DFSAccessAPIImpl();
    .........
 }

这个类实例化在内部使用 hadoop 修改的 jar,它不是从我的类加载器中获取的,它会抛出 ClassNotFoundException

我做错了什么?

这里使用的 JarClassLoader 是 jcloader : org.xeustechnologies.jcl.JarClassLoader

【问题讨论】:

  • 检查使用的 ClassLoader:Class.getClassLoader() 在不同的点和不同的类以验证它是您使用的类加载器。
  • @Mr_Thorynque : UserMigThreadImpl.class.getClassLoader() 显示原始父类加载器及其类,而不是我刚刚创建的新类加载器。如何解决这个问题?
  • 为了解决此类问题,创建了 OSGI。所以正确的解决方案是在 OSGI 容器中运行......我知道这个解决方案对你来说可能不现实。
  • 为什么在创建 UserMigThreadImpl 对象后在 currentThread 中设置新的 jcl?为什么不使用基本的 java.lang.ClassLoader ?

标签: java classloader


【解决方案1】:

我在将插件加载到我的应用程序时遇到了这个问题,因此我决定尝试从路径中的所有 jar 中加载所有 .class 文件。也许从我的应用中截取的这段代码会对你有所帮助。

https://bitbucket.org/rsohlich/plagdetector/src/432b52f252ff7647221b7e91b08731bd9cbe2a70/PlagDetectorSpring/src/main/java/cz/sohlich/app/service/impl/PluginHolderImpl.java

【讨论】:

  • 感谢@RadekSohlich。但我有 2 个版本的 jar 有冲突。所以我不应该在这里加载所有的罐子吗?
  • 好吧,我们一直在尝试这个,我们的大部分失败都与获取/使用错误的 ClassLoader 有关。 (应用服务器不同,独立 jar 不同)。现在我的想法是使用 URLClassLoader ucl = URLClassLoader.newInstance(urls, Thread.currentThread().getContextClassLoader());当前线程的引用在哪里。不确定 Thread.currentThread().setContextClassLoader(jcl);正确地工作。在实验中,我们是热交换 jars,它只是工作,类加载器总是用新的替换旧的。
猜你喜欢
  • 2012-02-01
  • 2018-12-14
  • 2011-09-30
  • 2013-09-06
  • 1970-01-01
  • 2013-10-01
  • 2017-11-04
  • 1970-01-01
  • 1970-01-01
相关资源
最近更新 更多