2

使用 XSLT Compiled Transform 时,我无法将输出完全格式化,它总是会去除所有空格并且不是人类可读的形式。

但是,如果我通过 Visual Studio XSLT 调试器运行相同的转换,则输出会整齐地缩进。这是我的代码的样子:

XDocument xDoc = XDocument.Load(_xmlFilePath, LoadOptions.PreserveWhitespace);
var xslDoc = new XmlDocument();
xslDoc.Load(_xsltFilePath);
xslDoc.PreserveWhitespace = true;
using (Stream outputFile = File.Create(_outputFileName))
{
    XsltArgumentList xsltParameter = null;
    StringReader tr = new StringReader(xslDoc.OuterXml);
    XmlReader xr = new XmlTextReader(tr);
    XslCompiledTransform transform = new XslCompiledTransform();
    XsltSettings setting = new XsltSettings();

    setting.EnableScript = true;
    transform.Load(xr, setting, new XmlUrlResolver());
    transform.Transform(xDoc.CreateReader(), xsltParameter, outputFile);
}

转换将创建一个文本文件,Visual Studio 的 XSLT 调试器在保留输出格式方面有何不同?

我已经看过这个链接:XSLT Transform not indenting proper

当我输出到文本时,这不适用于我。我应该使用 Stream 以外的东西吗?

我的示例 XSLT:

  <?xml version="1.0" encoding="utf-8"?>
<xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform"
xmlns:msxsl="urn:schemas-microsoft-com:xslt" exclude-result-prefixes="msxsl">
<xsl:output method="text" encoding="UTF-8"/>

<xsl:template match="/">
    <xsl:apply-templates/>
</xsl:template>

<xsl:template match="root">
    <xsl:text>{Start:</xsl:text>
    <xsl:text disable-output-escaping="yes">&#xD;</xsl:text>
    <xsl:value-of select="Child1"/>
    <xsl:text disable-output-escaping="yes">&#xD;</xsl:text>
    <xsl:value-of select="Child2"/>
    <xsl:text disable-output-escaping="yes">&#xD;</xsl:text>
    <xsl:text>End:}</xsl:text>
</xsl:template>

<?xml version="1.0" encoding="utf-8"?>
    <root>
    <Child1>Value1</Child1>
    <Child2>value2</Child2>
    </root>

预期输出:

{Start:
 Value1
 value2
 End:}

使用 XSLT 编译转换的输出: {Start:Value1value2End:}

Visual Studio XSLT 调试器为我提供了预期的输出格式,但 XSLT Compiled 转换去除了所有空格和换行符。

4

1 回答 1

2

不幸的是,这实际上仍然不是一个完整的代码示例。请阅读我之前提供的链接以了解我的意思。

就是说,经过一番努力,我能够采取零碎的工作并创建一个独立的、独立的代码示例。我能够验证您的担忧,我可以通过将您对disable-output-escaping(不需要)的使用替换为需要的xml:space="preserve",来解决这个问题。

请注意,XSL 转换的一部分是首先生成一个表示您的输出的 XML 图。只有在完成之后才会创建实际的输出。在此过程中,独立的空白将从文档中删除。通过添加xml:space="preserve",您可以告诉处理代码不要删除该独立空格。

这是您想要的 XSL 版本:

<?xml version="1.0" encoding="utf-8"?>
<xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform"
        xmlns:msxsl="urn:schemas-microsoft-com:xslt" exclude-result-prefixes="msxsl">
    <xsl:output method="text" encoding="UTF-8"/>

    <xsl:template match="/">
        <xsl:apply-templates/>
    </xsl:template>

    <xsl:template match="root">
        <xsl:text>{Start:</xsl:text>
        <xsl:text xml:space="preserve">&#xA;</xsl:text>
        <xsl:value-of select="Child1"/>
        <xsl:text xml:space="preserve">&#xA;</xsl:text>
        <xsl:value-of select="Child2"/>
        <xsl:text xml:space="preserve">&#xA;</xsl:text>
        <xsl:text>End:}</xsl:text>
    </xsl:template>
</xsl:stylesheet>

笔记:

  • 我不得不添加</xsl:stylesheet>结束标签
  • 我将 CR 实体更改&#xD;为 LF 实体&#xA;。它也可以像您一样使用回车,但是从可读性的角度来看,我觉得换行更“便携”。
  • 在上面的示例中,我应用xml:space="preserve"到每个xsl:text仅包含空格的元素。这可确保处理器不会删除这些元素。

最后一点需要权衡取舍。正如您将注意到的,这意味着您必须在xsl:text要保留空格的每个节点上指定该属性。另一种以 XSL 的可读性为代价的替代方法是将属性应用于整个样式表:

<?xml version="1.0" encoding="utf-8"?>
<xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform"
        xmlns:msxsl="urn:schemas-microsoft-com:xslt" exclude-result-prefixes="msxsl" xml:space="preserve">
    <xsl:output method="text" encoding="UTF-8"/>

    <xsl:template match="/"><xsl:apply-templates/></xsl:template>

    <xsl:template match="root"><xsl:text>{Start:</xsl:text><xsl:text>&#xD;</xsl:text><xsl:value-of select="Child1"/><xsl:text xml:space="preserve">&#xA;</xsl:text><xsl:value-of select="Child2"/><xsl:text xml:space="preserve">&#xA;</xsl:text><xsl:text>End:}</xsl:text></xsl:template>
</xsl:stylesheet>

显然,缺点是每个模板的内容必须在您希望它出现在输出中的位置包含空格。因此,您不能使用空格来使 XSL 本身更具可读性。但至少你不必xml:space="preserve"一遍又一遍地打字。你选。:)

我想从技术上讲,这意味着这毕竟不是 C# 问题。可能应该从您的问题中删除该标签。:)

于 2015-03-19T15:08:19.993 回答