DocumentBuilderFactory.setFeature调用失败的问题分析、解决javax.xml.parsers.DocumentBuilderFactory.setFeature(Ljava/lang/String;Z)V异常
mybatis启动报错
1. DocumentBuilderFactory加载顺序
- 使用 javax.xml.parsers.DocumentBuilderFactory 系统属性;
System.setProperty("javax.xml.parsers.DocumentBuilderFactory", "com.sun.org.apache.xerces.internal.jaxp.DocumentBuilderFactoryImpl");
- 使用 JRE 文件夹中的属性文件 "lib/jaxp.properties"。此配置文件格式为标准的 java.util.Properties 且包含实现类的完全限定名,其中实现类的键是上述定义的系统属性。 JAXP 实现只读取一次 jaxp.properties 文件,然后缓存其值供以后使用。如果首次尝试读取文件时,文件不存在,则不会再次尝试检查该文件是否存在。首次读取 jaxp.properties 后,其中的属性值不能再更改;
文件内容如下:
javax.xml.parsers.DocumentBuilderFactory=com.sun.org.apache.xerces.internal.jaxp.DocumentBuilderFactoryImpl
- 使用 Services API(在 JAR 规范中进行了详细描述)来确定类名称。Services API 将查找在运行时可用的 jar 中 META-INF/services/javax.xml.parsers.DocumentBuilderFactory 文件中的类名;
文件内容如下:
com.sun.org.apache.xerces.internal.jaxp.DocumentBuilderFactoryImpl
- 最后使用平台默认的 DocumentBuilderFactory 实例;
com.sun.org.apache.xerces.internal.jaxp.DocumentBuilderFactoryImpl
2. 引起问题的原因
项目引入了xercesImpl,在该jar中有Services API的定义,导致DocumentBuilderFactory初始化为org.apache.xerces.jaxp.DocumentBuilderFactoryImpl,而在这个实现类中没有setFeature方法
3. 尝试解决方法
- 项目中去除xercesImpl依赖
验证结果:无法解决
问题原因:好多地方有对xercesImpl的依赖,项目中去除对该JAR的直接依赖解决不了问题,如果完全去除会对其它功能有影响;
依赖xercesImpl的JAR
commons-dbcp
jaxen
xtom
- 在项目中增加Services API定义,把其设置为com.sun.org.apache.xerces.internal.jaxp.DocumentBuilderFactoryImpl
验证结果:无法解决
问题原因:实际还是使用了xercesImpl中定义的Services API
- 调整DocumentBuilderFactory的创建方式
DocumentBuilderFactory.newInstance("javax.xml.parsers.DocumentBuilderFactory", "com.sun.org.apache.xerces.internal.jaxp.DocumentBuilderFactoryImpl");
验证结果:问题解决
- 升级xercesImpl为有setFeature方法的高版本