【问题标题】:How to include two different versions of the same dependency?如何包含相同依赖项的两个不同版本?
【发布时间】:2014-11-17 07:46:34
【问题描述】:

我正在用 Java 定制一个 ERP 系统。在我的定制中,我想使用 Apache POI 3.10.1。因此我集成了罐子 poi-3.10.1-20140818.jar 和 poi-ooxml-3.10.1-20140818.jar。

但是,这些 jar 中包含的几个类已经包含在 ERP 系统的核心代码中,但有区别。

如果核心 ERP 类覆盖 POI 类,则自定义会引发运行时异常。如果 POI 类覆盖核心类,核心功能可能也会发生同样的情况。

处理此类问题的最佳做法是什么?

我的自定义是一个相对孤立的功能。

【问题讨论】:

    标签: java apache-poi classloader conflicting-libraries


    【解决方案1】:

    解决这个问题有两种方法:

    1. 您可以将库与加载其他 POI 版本的 ClassLoader 隔离开来。目前,我假设 ERP 系统位于类路径上,因此您需要将库与系统类加载器隔离开来。您可以通过创建URLClassLoader 的新实例来实现,然后将其指向包含较新版本 POI 的 jar 文件。确保还添加 all transient dependencies 例如 commons-codec 以避免类加载问题。另外,请注意瞬态依赖关系本身可以具有瞬态依赖关系。

      为了对类加载器隐藏类路径,您可以将引导类加载器设置为直接父级,由null 表示:

      new URLClassLoader(new URL[]{ new URL("poi-3.10.1-20140818.jar"), ... }, null);
      

      使用这个类加载器,您可以通过类似的方式查询较新版本的 POI 类

      Class.forName("org.apache.poi.hssf.usermodel.HSSFWorkbook", true, urlClassLoader);
      

      用于检索HSSFWorkbook 的新版本。但是请注意,任何直接引用 HSSFWorkbook 的文字都将由执行类的类加载器解析,这当然会链接旧的、不兼容的类版本。因此,您需要对所有代码使用反射。或者,您向 URLCLassLoader 添加一个包含所有逻辑的类,并且仅通过反射调用该类。一般来说,这是一种更清洁的方法。例如,您可以添加一个实现引导类的类,例如 Callable,然后您可以在任何不同的上下文中使用它,例如:

      Callable<File> sub = (Callable<File>) Class.forName("pkg.Subroutine", 
                                                          true, 
                                                          urlClassLoader);
      File convertedFile = sub.call();
      
    2. 或者,您可以将第二个 POI 依赖项重新打包到另一个名称空间中。完成此操作后,类不再冲突,因为它们的名称不再相等。这可能是一种更简洁的方法,因为您可以使用同一个类加载器中的两个库并避免反射。

      为了将依赖项重新打包到另一个名称空间中,有像 Maven Shade plugin 这样的工具可以帮助您完成这项任务。替代方案是jarjar 对应antShadow plugin 对应Gradle

    【讨论】:

      【解决方案2】:

      如果您使用的是 Servlet 3.0 API 并且您可以更改一些配置,那么“web 片段”可以用于这种情况。以下是解释:http://www.oracle.com/technetwork/articles/javaee/javaee6overview-part2-136353.html#webfrags

      【讨论】:

        猜你喜欢
        • 1970-01-01
        • 2020-02-07
        • 2013-05-23
        • 1970-01-01
        • 1970-01-01
        • 1970-01-01
        • 2019-12-27
        • 2011-09-02
        • 2016-05-24
        相关资源
        最近更新 更多