<root>
<element>
<id>1</id>
<group>first</group>
</element>
<element>
<id>2</id>
<group>second</group>
</element>
<element>
<id>3</id>
<group>first</group>
</element>
...
<root>
Run Code Online (Sandbox Code Playgroud)
我如何按xslt 1.0中的组名对元素进行分组.输出:
<root>
<group name="first">
<element>
<id>1</id>
<group>first</group>
</element>
<element>
<id>3</id>
<group>first</group>
</element>
</group>
<group name="second">
<element>
<id>2</id>
<group>second</group>
</element>
</group>
</root>
Run Code Online (Sandbox Code Playgroud)
有任何想法吗?
Tim*_*m C 15
这是Muenchian Grouping的工作.您将在StackOverflow上的XSLT标记中找到许多示例.
首先,您需要定义一个键来帮助您对组进行分组
<xsl:key name="groups" match="group" use="."/>
Run Code Online (Sandbox Code Playgroud)
这将查找给定组名的组元素.
接下来,您需要匹配每个distince组名称的第一个实例的所有匹配项.这是通过这个可怕的陈述完成的
<xsl:apply-templates select="element/group[generate-id() = generate-id(key('groups', .)[1])]"/>
Run Code Online (Sandbox Code Playgroud)
即匹配组元素恰好是我们的键中该元素的第一个出现.
当您匹配不同的组节点时,您可以循环遍历具有相同名称的所有其他组节点(其中$ currentGroup是包含当前组名的变量)
<xsl:for-each select="key('groups', $currentGroup)">
Run Code Online (Sandbox Code Playgroud)
完全放弃这个
<xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
<xsl:key name="groups" match="group" use="."/>
<xsl:template match="/root">
<root>
<xsl:apply-templates select="element/group[generate-id() = generate-id(key('groups', .)[1])]"/>
</root>
</xsl:template>
<xsl:template match="group">
<xsl:variable name="currentGroup" select="."/>
<group>
<xsl:attribute name="name">
<xsl:value-of select="$currentGroup"/>
</xsl:attribute>
<xsl:for-each select="key('groups', $currentGroup)">
<element>
<id>
<xsl:value-of select="../id"/>
</id>
<name>
<xsl:value-of select="$currentGroup"/>
</name>
</element>
</xsl:for-each>
</group>
</xsl:template>
</xsl:stylesheet>
Run Code Online (Sandbox Code Playgroud)
将此应用于您的示例XML会产生以下结果
<root>
<group name="first">
<element>
<id>1</id>
<name>first</name>
</element>
<element>
<id>3</id>
<name>first</name>
</element>
</group>
<group name="seccond">
<element>
<id>2</id>
<name>seccond</name>
</element>
</group>
</root>
Run Code Online (Sandbox Code Playgroud)
Dim*_*hev 15
I.这是一个完整且非常简短的XSLT 1.0解决方案:
<xsl:stylesheet version="1.0"
xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
<xsl:output omit-xml-declaration="yes" indent="yes"/>
<xsl:key name="kElsByGroup" match="element" use="group"/>
<xsl:template match="/*">
<root>
<xsl:apply-templates/>
</root>
</xsl:template>
<xsl:template match=
"element[generate-id()=generate-id(key('kElsByGroup',group)[1])]">
<group name="{group}">
<xsl:copy-of select="key('kElsByGroup',group)"/>
</group>
</xsl:template>
<xsl:template match=
"element[not(generate-id()=generate-id(key('kElsByGroup',group)[1]))]"/>
</xsl:stylesheet>
Run Code Online (Sandbox Code Playgroud)
当此转换应用于提供的XML文档时:
<root>
<element>
<id>1</id>
<group>first</group>
</element>
<element>
<id>2</id>
<group>second</group>
</element>
<element>
<id>3</id>
<group>first</group>
</element>
</root>
Run Code Online (Sandbox Code Playgroud)
产生了想要的正确结果:
<root>
<group name="first"><element>
<id>1</id>
<group>first</group>
</element><element>
<id>3</id>
<group>first</group>
</element></group>
<group name="second"><element>
<id>2</id>
<group>second</group>
</element></group>
</root>
Run Code Online (Sandbox Code Playgroud)
请注意:
使用Muenchian方法进行分组.这是XSLT 1.0中最有效的分组方法.
使用AVT(属性值模板)将文字结果元素及其变量 - 值属性指定为一个整体.使用AVT简化了编码并产生更短且更易理解的代码.
II.更短的XSLT 2.0解决方案:
<xsl:stylesheet version="2.0"
xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
<xsl:output omit-xml-declaration="yes" indent="yes"/>
<xsl:template match="/*">
<root>
<xsl:for-each-group select="element" group-by="group">
<group name="{current-grouping-key()}">
<xsl:copy-of select="current-group()"/>
</group>
</xsl:for-each-group>
</root>
</xsl:template>
</xsl:stylesheet>
Run Code Online (Sandbox Code Playgroud)
当此转换应用于同一XML文档(上面)时,会再次生成相同的正确结果.
请注意:
0.1.使用<xsl:for-each-group> XSLT 2.0指令.
0.2.使用标准的XSLT 2.0函数current-group()和current-grouping-key()