如何包含同一依赖项的两个不同版本?
我正在使用Java进行ERP系统的定制。 在我的自定义中,我想使用Apache POI 3.10.1。 因此我整合了jar子poi-3.10.1-20140818.jar和poi-ooxml-3.10.1-20140818.jar。
但是,这些jar包含已经包含在ERP系统核心代码中的几个类,但有差异。
如果核心ERP类覆盖POI类,则自定义会引发运行时exception。 如果POI类覆盖核心类,则核心function可能会发生相同的情况。
处理这样的问题的最佳做法是什么?
我的定制是一个相对孤立的function。
有两种方法可以解决这个问题:
-
您可以将库与加载其他版本POI的
ClassLoader
隔离。 现在,我假设ERP系统位于类路径上,因此您需要将库与系统类加载器隔离开来。 您可以通过创建URLClassLoader
的新实例来执行此操作,然后指向包含较新版本POI的jar文件。 确保还添加所有瞬态依赖项 ,例如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
的任何直接引用都HSSFWorkbook
执行类的类加载器解决,这当然会链接类的旧的,不兼容的版本。 因此,您需要对所有代码使用reflection。 或者,您可以向URLCLassLoader
添加一个类,其中包含您的所有逻辑,并且只能通过reflection调用此类。 总的来说,这是一种更清洁的方法。 例如,您可以添加一个实现引导类的类,例如Callable
,然后您可以从任何不同的上下文中使用它,例如:Callable
sub = (Callable ) Class.forName("pkg.Subroutine", true, urlClassLoader); File convertedFile = sub.call(); -
或者,您可以将第二个POI依赖项重新打包到另一个名称空间中。 执行此操作后,类不再冲突,因为它们的名称不再相等。 这可能是一种更简洁的方法,因为您可以使用来自同一类加载器的两个库,并避免reflection。
要将依赖项重新打包到另一个名称空间,可以使用Maven Shade插件等工具来帮助您完成此任务。 替代品是用于ant的 jarjar或用于Gradle的Shadow插件 。
如果您使用的是Servlet 3.0 API并且可以更改某些配置,则可以将“Web片段”用于此类情况。 以下是解释: http : //www.oracle.com/technetwork/articles/javaee/javaee6overview-part2-136353.html#webfrags
- 使用Apache POI读取xlsx文件时获取exception(org.apache.poi.openxml4j.exception – 无内容类型)?
- 使用页码指示为XWPFDocument创建目录
- 是否有支持Microsoft Office和Open Office的Java库?
- 如何避免Apache POI中的java.lang.NoSuchMethodError:org.apache.poi.util.IOUtils.copy(Ljava / io / InputStream; Ljava / io / OutputStream;)
- Apache POI工作簿的默认样式
- 如何使用Apache POI创建简单的docx文件?
- 使用Apache POI生成Excel下拉列表时的限制
- 在LibreOffice中不会自动调整Apache poi Excel行高
- 如何使用Apache POI操纵注释的内容