我正在为Java中的ERP系统进行定制。在我的定制中,我想使用Apache POI 3.10.1。因此,我集成了jars poi-3.10.1-20140818.jar和poi-ooxml-3.10.1-20140818.jar。
但是,这些jar包含几个类,这些类已经包含在ERP系统的核心代码中,但是有所不同。
如果核心ERP类覆盖POI类,则定制将引发运行时异常。如果POI类覆盖核心类,则核心功能可能会发生相同的情况。
解决此类问题的最佳实践是什么?
我的自定义是一个相对隔离的功能。
有两种方法可以解决此问题:
ClassLoader
URLClassLoader
为了对类加载器隐藏类路径,您可以将bootstrap类加载器设置为直接父级,由表示null:
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,然后您可以在任何其他上下文中使用该类,例如:
HSSFWorkbook
URLCLassLoader
Callable
Callable<File> sub = (Callable<File>) Class.forName("pkg.Subroutine", true, urlClassLoader); File convertedFile = sub.call();
要将依赖关系重新打包到另一个名称空间中,可以使用Maven Shade插件之类的工具来帮助您完成此任务。替代方案是jarjar为蚂蚁或阴影插件的摇篮。