如何包含同一依赖项的两个不同版本



我正在用Java为ERP系统进行定制。在我的自定义中,我想使用Apache POI 3.10.1。因此,我集成了poi-3.10.1-20140818.jar和poi-ooxml-3.10.1-20110818.jar。

然而,这些jar包含几个类,这些类已经包含在ERP系统的核心代码中,但存在差异。

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

处理这样的问题的最佳实践是什么?

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

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

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

    为了向类加载器隐藏类路径,您可以将引导类加载器设置为直接父类,由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);
    

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

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

    对于将依赖项重新打包到另一个名称空间,有一些工具,如Maven Shade插件,可以帮助您完成此任务。替代方案是ant的jarjar或Gradle的Shadow插件。

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

相关内容

  • 没有找到相关文章

最新更新