我有一个动态XML文档,它表示类别的树结构,但使用路径分隔的属性以任意顺序 - 这样:
<data>
<record ID="24" Name="category 1\sub category 1"/>
<record ID="26" Name="category 1"/>
<record ID="25" Name="category 1\sub category 1\sub category 2"/>
<record ID="27" Name="category 1\sub category 1\sub category 3"/>
...
</data>
我需要提出一个“规范化”XML的解决方案,以便将其转换为以下内容:
<data>
<record ID="26" Name="category 1">
<record ID="24" Name="sub category 1">
<record ID="25" Name="sub category 2"/>
<record ID="27" Name="sub category 3"/>
</record>
</record>
...
</data>
基本上我想知道这是XSLT可以解决的问题,以及如何,而不是必须以编程方式进行。
答案 0 :(得分:18)
当然,没问题:
<xsl:stylesheet
version="1.0"
xmlns:xsl="http://www.w3.org/1999/XSL/Transform"
>
<xsl:output indent="yes" />
<xsl:template match="/data">
<!-- copy the document element -->
<xsl:copy>
<!-- That's where we start: all "record" nodes that have no "\". -->
<xsl:apply-templates mode="recurse" select="/data/record[
not(contains(@Name, '\'))
]" />
</xsl:copy>
</xsl:template>
<xsl:template match="record" mode="recurse">
<xsl:param name="starting-path" select="''" />
<!-- The record node and its ID attribute can be copied. -->
<xsl:copy>
<xsl:copy-of select="@ID" />
<!-- Create the new "name" attribute. -->
<xsl:attribute name="Name">
<xsl:value-of select="substring-after(@Name, $starting-path)" />
</xsl:attribute>
<!-- Append a backslash to the current path. -->
<xsl:variable name="current-path" select="concat(@Name, '\')" />
<!-- Select all "record" nodes that are one level deeper... -->
<xsl:variable name="children" select="/data/record[
starts-with(@Name, $current-path)
and
not(contains(substring-after(@Name, $current-path), '\'))
]" />
<!-- ...and apply this template to them. -->
<xsl:apply-templates mode="recurse" select="$children">
<xsl:with-param name="starting-path" select="$current-path" />
</xsl:apply-templates>
</xsl:copy>
</xsl:template>
</xsl:stylesheet>
我的系统输出:
<data>
<record ID="26" Name="category 1">
<record ID="24" Name="sub category 1">
<record ID="25" Name="sub category 2"></record>
<record ID="27" Name="sub category 3"></record>
</record>
</record>
</data>
请注意,整个解决方案基于以下假设:所有路径都是规范的,并且不包含尾部反斜杠。
另请注意,任何不匹配/孤立的“记录”元素都不会出现在输出中(当然,除非它们位于根级别)。
还有一件事:模板模式(“递归”)并非绝对必要。我包含它是因为模板正在做一些相当特殊的事情,并且您的解决方案中可能存在与“记录”节点匹配的另一个模板。在这种情况下,可以放入此解决方案而不会破坏任何其他内容。对于独立解决方案,可以随时删除模板模式。
哦,最后一件事:如果您希望按名称排序结果文档,请添加<xsl:sort>
element <xsl:apply-templates>
(两次出现),如下所示:
<xsl:apply-templates select="...">
<xsl:sort select="@Name" data-type="text" order="ascending" />
</xsl:apply-templates>