寻找在动态加载 Jar 文件和在 Java 运行时实例化类时使用 Apache Felix 的基本示例

Posted

技术标签:

【中文标题】寻找在动态加载 Jar 文件和在 Java 运行时实例化类时使用 Apache Felix 的基本示例【英文标题】:Looking for basic example of using Apache Felix in dynamic loading of Jar file and instancing a class at runtime in Java 【发布时间】:2012-07-16 23:19:35 【问题描述】:

我尝试根据一些示例实现自己的类加载器。但是,我认为它没有按预期工作(无法重新加载 Jar 文件等。我看到很少有推荐使用 OSGI 或 Apache Felix 来处理 Jar 文件加载的参考资料。有没有加载 Jar 的示例,从 Jar 实例化一个类?

关于我要完成的工作的更多详细信息..我有一个基本上连续运行的 Java 命令行应用程序。我希望它能够在运行时动态引用 JAR 文件,并且在运行时实例中这些 jar 中的类。这些 jar 可能包含 1 个或多个支持类。这些 Jars 本质上是定制的工作单元,由持续运行的主应用程序的某些事件条件执行。 .. 由于这是一个多客户端,我想让 jar 成为可插入类型的工作单元。

我目前的方向是为“客户端”提供一个代码接口,然后让他们将自己的类打包到一个 jar 文件中。然后应用程序将执行配置的(数据库)jar 路径并从 Jar 运行类。这适用于加载 jar 和类但是,我希望能够基本上热部署这些 jar。

总之,我想要一个 JAR 文件,其中包含支持某个功能的类。将在运行时从 Jar(来自主循环应用程序)引用的已定义类。在主应用程序运行时更改 JAR 文件的能力。

如果我要使用第三方库,我最好使用 Apache Felix。

谢谢

我想我是使用 Apache Felix 4 解决的。这是加载 jars/classes 的最佳方式吗?或者有没有更好更有效的方法。到目前为止,我的研究并没有指向一种解决方案。谢谢。

    FrameworkFactory ff = new FrameworkFactory ();
     Map<String,Object> config = new HashMap<String,Object>();
      config.put("org.osgi.framework.storage", "c:\\temp\\myclientBundles");
       config.put(Constants.FRAMEWORK_SYSTEMPACKAGES_EXTRA,
   "packages needed,more packages needed");
  config.put(Constants.FRAMEWORK_STORAGE_CLEAN, "true");

  Framework fwk = ff.newFramework(config);
  fwk.start();


  BundleContext bc = fwk.getBundleContext();

  Bundle bundle =   bc.installBundle("file:C:\\my_client.jar");


      bundle.start();
      bundle.update();
      Class clazz = bundle.loadClass("client.work.process");
      Job pc = (Job) clazz.newInstance();
      pc.startWork(someConfigObject);
      bundle.stop();
      fwk.stop();

【问题讨论】:

问题:我必须将类所需的所有包添加到 FrameWorkFactory 和 bundle jar 清单文件中吗?有没有更好的方法来处理这个问题?通配符不起作用。我的意思是我正在使用的 jars/bundle 有几个依赖项。 【参考方案1】:

如果现在,简单地说,我拒绝相信你的问题是加载一个类... :-) 我认为你还有另一个问题,你认为加载一个类可以解决?

在大多数情况下,人们天真地开始加载类,问题在于可扩展性。他们有一个系统并希望有选择地使用新功能对其进行扩展,我假设您有类似的问题,因为您想更新提供程序 jar?

如果是这样,请下载 bndtools 并查看 OSGi 服务,它们通常非常符合要求

好的,更新后。如果我很了解您,那么 Apache Felix 和 Apache Felix File Install 将为您提供很好的服务。 File Install 监视一个目录并在框架中的该目录中安装任何包。从目录中删除该 jar,将卸载该捆绑包。 (我在 12 年前编写了 File Install 的原型!)

对于你的主 JAR,让它看起来像:

@Component(immediate=true)
public void Main   
    @Reference
    void setClient( Client client)  ...  // called whenever a client gets started

对于每个客户端 JAR:

@Component
public void ClientImpl implements Client 
  ... whatever

这几乎是您使用 bndtools 时必须编写的全部内容。只需创建一个组件项目,为主代码添加一个 Bundle Descriptor 并为客户端代码示例添加任意数量的描述符,然后 Run As -> OSGi Run。然后,您可以下载 Apache Felix 文件安装,将其添加到运行选项卡,并创建一些其他项目,将 jars(在生成的文件夹中连续制作)放入文件安装文件夹中......瞧。没有比这更简单的了:-)

【讨论】:

感谢您的回复。对于主应用程序,我不需要通过启动OSGI框架并获取相应的类实例的过程吗?对于客户端 jar,我正在学习 bndtools 插件。我越来越熟悉,但不确定如何“发布”或提供捆绑包的执行方法。我可能有点偏离轨道,但我更接近几天前的位置。再次感谢您的帮助 有两种方法。你可以保持你的主应用程序不变并嵌入一个 OSGi 框架(Apache Felix 有一个很好的教程,它也适用于其他框架:felix.apache.org/site/…)。如果您在该嵌入式框架中安装 FileInstall,您会自动在加载(或工作)目录中加载任何包(和配置)。如果这些注册服务,您可以使用主应用程序中的服务跟踪器跟踪它们,这会在服务注册时为您提供回调。否则,请将您的应用程序打包。【参考方案2】:

如果我理解正确,您是在询问如何在 OSGi 中动态加载 jar 的示例,对吗?

在 OSGi 中,动态加载的模块称为捆绑包。 因此,基本上解决您的问题的最常用方法是将您的 jar 放入 OSGi 捆绑包中。 OSGi 包是一个普通的 Java jar,有一些增强 - 它有一个清单文件(一个 txt 文件),其中声明了对其他 Java 包的依赖关系;以及一个 Activator 类,它在包启动和停止时被调用。这个 Activator 是包的核心部分,它知道如何实例化包的其他部分。

现在,如果您的动态 jar 只是一个库 jar,并且您只想在系统中提供它的 java 包,您甚至可以不使用激活器,只需添加一个简单的清单文件。甚至还有一些工具可以为您自动进行转换 - 请查看 BND。

然后,当您准备好捆绑包时,您可以使用 install 命令在任何 OSGi 框架上动态安装它。重新加载是通过调用更新命令等来完成的。生命周期定义得很好,我肯定会推荐使用它而不是重新发明***;)

有关示例和演示,请参阅here。

希望这会有所帮助。

【讨论】:

我确实看到了这个 BNDtools 非常有用的地方。关于将 jar 文件转换为捆绑包的任何好的教程?我一直在使用遵循某种模式的 jar 和遵循接口的类。我对 BNDTools 和捆绑包构建的了解仍然是基础知识,因此感谢您提供任何信息。谢谢

以上是关于寻找在动态加载 Jar 文件和在 Java 运行时实例化类时使用 Apache Felix 的基本示例的主要内容,如果未能解决你的问题,请参考以下文章

在加载类时,如何指定所使用的jar包

关于java动态加载jar的问题?

如何在Java运行的时候动态加载一个jar包到classpath里面

JAVA怎么动态加载JAR包

Java_Java中动态加载jar文件和class文件

如何在模块化的 java 11 应用程序中动态加载 Libreoffice jar,而不从自定义类加载器中获取 ClassCastException