我尝试使用 Saxon 代替 JDK 的默认实现(我猜是 Xalan)来进行 XML 转换和 Xpath。在我的代码中,我使用 document.createCDATASection(data) 方法创建一个 CDATA 节点。代码如下所示:
CDATASection cdata = doc.createCDATASection("data");
Node valueNode = node.appendChild(doc.createElement("value"));
valueNode.appendChild(cdata);
其中节点是我的 XML 中的某个随机节点。 它与 JDK 的默认实现配合得很好,生成的 XML 如下所示:
<node>
<value><![CDATA[data]]></value>
</node>
如果我包含 Saxon maven 工件(请注意,它只是包含,工厂选择/实例化是默认的,就像之前一样),并且所有 cdata 节点都被视为简单文本节点,即 XML 变为:
,相同的代码开始表现奇怪 <node>
<value>data</value>
</node>
在检索时会导致问题,因为该代码专门检查 cdata 元素,这些元素在后面的情况下已被删除。我不确定为什么会发生这种情况(看起来我没有正确使用它)。我还尝试从我的 POM(Saxon 的传递依赖)中排除 Xerces 工件,但没有成功。此外,还验证了 JDK 本身正在使用 DocumentBuilderFactory 等的实现类。如果我做错了什么,请专家帮助我。
提前致谢。
我猜您的应用程序可能正在执行从 DOMSource 到 StreamResult 的 JAXP 身份转换,以便序列化 DOM。 JAXP 身份转换的 Saxon 实现使用 XSLT 的序列化规则,这具有删除 CDATA 部分的效果。这完全符合 JAXP,即使它不是默认的 JDK 实现的做法。
如果您依赖于 JAXP 身份转换器的特定实现的行为,那么您不应该编写应用程序来获取恰好位于类路径上的任何实现;你应该明确地实例化你想要的实现。
如果调用身份转换的代码不是您自己编写的并且无法轻易更改,那么这当然会很困难。在这种情况下,最好的方法是设置系统属性 javax.xml.transform.TransformerFactory 以选择 Xalan,并且在您想要调用 Saxon 的地方显式执行,而不是依赖于 JAXP 工厂搜索。
我也有一个类似的问题,并且已从传递依赖项中排除了 saxon jar。我在本地运行并测试,但它没有出现在服务器中。我应该寻找什么额外的东西来在服务器中调试