DocumentBuilderFactory.setFeature调用失败的问题分析、解决javax.xml.parsers.DocumentBuilderFactory.setFeature(Lja

DocumentBuilderFactory.setFeature调用失败的问题分析、解决javax.xml.parsers.DocumentBuilderFactory.setFeature(Lja

 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方法

 DocumentBuilderFactory.setFeature调用失败的问题分析、解决javax.xml.parsers.DocumentBuilderFactory.setFeature(Lja

 

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方法的高版本
上一篇:通配符的匹配很全面, 但无法找到元素 ‘context:component-scan’ 的声明


下一篇:c-隐藏符号时dynamic_cast失败