我有一个由转换生成的XdmNode对象,我需要写入包含一些CDATA的流,但是我无法获得包含CDATA转义的输出。
这是因为我没有经历序列化过程,例如,使用Serializer类吗?如果是这样,我该怎么做?我可以看到Serializer类在Java中有一个serializeNode()方法,但在C#中却没有...只有转换/ etc似乎能够“使用”它。或者这是围绕cdata-section-elements语句的问题吗?
使用此处的XML和XSLT来说明: How do I force xslt transformation to load data into cdata sections?
这在C#中使用.NET 4.5和Saxon 9.6.0.6。
C#代码:
Processor processor = new Processor();
XdmNode node = GetNode(processor); //gets XdmNode for XML doc below
XsltTransformer transformer = GetTransformer(processor); //gets transform below
transformer.InitialContextNode = node;
XdmDestination output = new XdmDestination();
transformer.Run(output);
string results = output.XdmNode.OuterXml;
using (XmlWriter writer = XmlWriter.Create(Console.Out))
{
output.XdmNode.WriteTo(writer);
}
XSLT:
<xsl:stylesheet version="1.0"
xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
<xsl:output omit-xml-declaration="yes" indent="yes"
cdata-section-elements="num"/>
<xsl:strip-space elements="*"/>
<xsl:template match="node()|@*">
<xsl:copy>
<xsl:apply-templates select="node()|@*"/>
</xsl:copy>
</xsl:template>
</xsl:stylesheet>
XML:
<nums>
<num>01</num>
<num>02</num>
<num>03</num>
<num>04</num>
<num>05</num>
<num>06</num>
<num>07</num>
<num>08</num>
<num>09</num>
<num>10</num>
</nums>
预期产出:
<nums>
<num><![CDATA[01]]></num>
<num><![CDATA[02]]></num>
<num><![CDATA[03]]></num>
<num><![CDATA[04]]></num>
<num><![CDATA[05]]></num>
<num><![CDATA[06]]></num>
<num><![CDATA[07]]></num>
<num><![CDATA[08]]></num>
<num><![CDATA[09]]></num>
<num><![CDATA[10]]></num>
</nums>
实际输出(在控制台和字符串中):
<nums>
<num>01</num>
<num>02</num>
<num>03</num>
<num>04</num>
<num>05</num>
<num>06</num>
<num>07</num>
<num>08</num>
<num>09</num>
<num>10</num>
</nums>
答案 0 :(得分:1)
似乎您要做的是将XdmNode发送到Serializer,使其序列化,并使用Serializer上设置的属性。最简单的方法是运行虚拟查询:
QueryCompiler qc = Processor.NewQueryCompiler();
QueryEvaluator qe = qc.Compile(".").Load();
qe.ContextItem = xdmNode;
qe.Run(serializer);
XQuery表达式“。”只需返回上下文项。
顺便提一下,Serializer的API文档没有说明CDATA_SECTION_ELEMENTS属性应该采用什么形式,但我认为它可能是Clark表示法中以空格分隔的QNames序列,即Q{uri}local
。如果没有名称空间,则只是本地名称。
(这类似于运行“身份转换”的Java JAXP接口中经常使用的方法。但身份查询要简单得多。)