我有以下输入:
<articles>
<item name="B">
<item name="Bb">
<item name="Bbb"/>
<item name="Abb"/>
</item>
<item name="Ab">
<item name="Bab"/>
<item name="Aab"/>
</item>
</item>
<item name="A">
<item name="Ba">
<item name="Bba"/>
<item name="Aba"/>
</item>
<item name="Aa">
<item name="Baa"/>
<item name="Aaa"/>
</item>
</item>
<item name="D"/>
<item name="C">
<item name="Ac"/>
</item>
</articles>
需要以下输出:
A-Aa-Aaa
A-Aa-Baa
A-Ba-Aba
A-Ba-BBa
B-Ab-Aab
B-Ab-Bab
B-Bb-Abb
B-Bb-Bbb
C-Ac
D
也就是说,每条路径都应该是明确的,并在每个级别上进行排序。我有以下XSLT,它产生路径,但我无法弄清楚排序,除了第一级。
<xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
<xsl:output method="text"/>
<xsl:strip-space elements="*"/>
<xsl:template match="articles">
<xsl:apply-templates select="//item[not(item)]">
<xsl:sort select="ancestor-or-self::item[parent::articles]/@name"/>
</xsl:apply-templates>
</xsl:template>
<xsl:template match="item">
<xsl:for-each select="ancestor-or-self::item">
<xsl:value-of select="@name"/>
<xsl:choose>
<xsl:when test="not(item)">
<xsl:text>
</xsl:text>
</xsl:when>
<xsl:otherwise>-</xsl:otherwise>
</xsl:choose>
</xsl:for-each>
</xsl:template>
</xsl:stylesheet>
我担心目前的做法是死路一条。想法?
答案 0 :(得分:3)
请注意,XSLT 2.0解决方案(第二部分)更短更简单。
<强>予。这是一个简单的2遍解决方案:
<xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform"
xmlns:ext="http://exslt.org/common">
<xsl:output method="text"/>
<xsl:strip-space elements="*"/>
<xsl:template match="/">
<xsl:variable name="vrtfPass1">
<xsl:apply-templates/>
</xsl:variable>
<xsl:for-each select="ext:node-set($vrtfPass1)/*">
<xsl:sort/>
<xsl:value-of select="concat(., '
')"/>
</xsl:for-each>
</xsl:template>
<xsl:template match="item[not(item)]">
<x>
<xsl:for-each select="ancestor-or-self::item">
<xsl:if test="not(position() = 1)">-</xsl:if>
<xsl:value-of select="@name"/>
</xsl:for-each>
</x>
</xsl:template>
</xsl:stylesheet>
在提供的XML文档上应用此转换时:
<articles>
<item name="B">
<item name="Bb">
<item name="Bbb"/>
<item name="Abb"/>
</item>
<item name="Ab">
<item name="Bab"/>
<item name="Aab"/>
</item>
</item>
<item name="A">
<item name="Ba">
<item name="Bba"/>
<item name="Aba"/>
</item>
<item name="Aa">
<item name="Baa"/>
<item name="Aaa"/>
</item>
</item>
<item name="D"/>
<item name="C">
<item name="Ac"/>
</item>
</articles>
产生了想要的正确结果:
A-Aa-Aaa
A-Aa-Baa
A-Ba-Aba
A-Ba-Bba
B-Ab-Aab
B-Ab-Bab
B-Bb-Abb
B-Bb-Bbb
C-Ac
D
请注意:这是一个通用解决方案,可以正确使用 item
元素的任何嵌套。
<强>解释强>:
我们进行两次转换。
第一遍的结果是:
<x xmlns:ext="http://exslt.org/common">B-Bb-Bbb</x>
<x xmlns:ext="http://exslt.org/common">B-Bb-Abb</x>
<x xmlns:ext="http://exslt.org/common">B-Ab-Bab</x>
<x xmlns:ext="http://exslt.org/common">B-Ab-Aab</x>
<x xmlns:ext="http://exslt.org/common">A-Ba-Bba</x>
<x xmlns:ext="http://exslt.org/common">A-Ba-Aba</x>
<x xmlns:ext="http://exslt.org/common">A-Aa-Baa</x>
<x xmlns:ext="http://exslt.org/common">A-Aa-Aaa</x>
<x xmlns:ext="http://exslt.org/common">D</x>
<x xmlns:ext="http://exslt.org/common">C-Ac</x>
第一遍的处理使用仅匹配 leaf item
元素的单个模板。对于每个叶item
元素,其完整的划线分隔路径将作为名为x
的元素的内容输出。
第二遍只是对第一遍的结果进行排序。
<强> II。 XSLT 2.0解决方案:
<xsl:stylesheet version="2.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
<xsl:output method="text"/>
<xsl:template match="/">
<xsl:for-each-group select="//item[not(item)]"
group-by="string-join(ancestor-or-self::item/@name, '-')">
<xsl:sort select="current-grouping-key()"/>
<xsl:sequence select="current-grouping-key(), '
'"/>
</xsl:for-each-group>
</xsl:template>
</xsl:stylesheet>
答案 1 :(得分:0)
不漂亮,但以下是针对3个项目的深度,并且每个附加级别需要额外的排序。我确信一位大师能够更一般地做到这一点:)
<xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
<xsl:output method="text"/>
<xsl:strip-space elements="*"/>
<xsl:template match="/">
<xsl:apply-templates/>
</xsl:template>
<xsl:template match="articles">
<xsl:apply-templates select="//item[not(item)]">
<xsl:sort select="ancestor-or-self::item[parent::articles]/@name"/>
<xsl:sort select="ancestor-or-self::item[parent::item[parent::articles]]/@name"/>
<xsl:sort select="ancestor-or-self::item[parent::item[parent::item[parent::articles]]]/@name"/>
</xsl:apply-templates>
</xsl:template>
<xsl:template match="item">
<xsl:for-each select="ancestor-or-self::item">
<xsl:value-of select="@name"/>
<xsl:choose>
<xsl:when test="not(item)">
<xsl:text>
</xsl:text>
</xsl:when>
<xsl:otherwise>-</xsl:otherwise>
</xsl:choose>
</xsl:for-each>
</xsl:template>
</xsl:stylesheet>