我有以下XML:
<League>
<Week Date="26/04/2010 19:00">
<Fixture Id="542" HomeTeamId="371" HomeTeam="London Raiders Green" AwayTeamId="13" AwayTeam="Richmond Swingers" />
<Fixture Id="543" HomeTeamId="45" HomeTeam="Spartans" AwayTeamId="15" AwayTeam="Panthers" />
<Fixture Id="544" HomeTeamId="370" HomeTeam="Fat Cats" AwayTeamId="381" AwayTeam="London Raiders Orange" />
</Week>
<Week Date="27/04/2010 19:00">
<Fixture Id="548" HomeTeamId="3" HomeTeam="The Mob" AwayTeamId="81" AwayTeam="London Raiders Red" />
<Fixture Id="549" HomeTeamId="373" HomeTeam="Intellect" AwayTeamId="83" AwayTeam="Tornadoes" />
</Week>
</League>
我想要的是获取该XML中所有团队ID的唯一列表,但问题是团队ID可以显示在HomeTeamId
或AwayTeamId
属性中夹具节点。所以我很难使用标准的分组方法(Grouping using the Muenchian method或selecting unique nodes by checking the preceding sibling)。
我可以通过这种方式获得所有ID的列表:
<xsl:for-each select="//Fixture/@HomeTeamId | //Fixture/@AwayTeamId">
<xsl:sort select="."/>
<xsl:value-of select="."/><br/>
</xsl:for-each>
但是当然当团队出现在多个灯具中时,他们的id会使用上面的for-each输出多次。
我的最终目标是输出由每个团队分组的灯具列表,但我在这里与XSLT斗争 - 它让我的大脑受伤......有人能指出我正确的方向吗?
答案 0 :(得分:4)
使用Muenchian分组应该不是问题:
<xsl:stylesheet xmlns:xsl="http://www.w3.org/1999/XSL/Transform" version="1.0">
<xsl:output indent="yes"/>
<xsl:key name="k1" match="Fixture/@HomeTeamId | Fixture/@AwayTeamId" use="."/>
<xsl:template match="/">
<html>
<head>
<body>
<xsl:for-each select="(//Fixture/@HomeTeamId | //Fixture/@AwayTeamId)[generate-id() = generate-id(key('k1', .)[1])]">
<xsl:sort select="." data-type="number"/>
<xsl:value-of select="."/>
<br/>
</xsl:for-each>
</body>
</head>
</html>
</xsl:template>
</xsl:stylesheet>
我不会使用for-each而是使用apply-templates但是因为你已经在样本中使用了for-each,所以我保留了这个以展示如何使用Muenchian分组。
答案 1 :(得分:2)
此转换会产生唯一的团队ID,即使它们的值属于任意数量的不同命名的属性:
<xsl:stylesheet version="1.0"
xmlns:xsl="http://www.w3.org/1999/XSL/Transform"
xmlns:my="my:my" exclude-result-prefixes="my">
<xsl:output omit-xml-declaration="yes" indent="yes"/>
<xsl:key name="kTeamById"
match="@*[name()=document('')/*/my:teamIdNames/*]" use="."/>
<my:teamIdNames>
<name>HomeTeamId</name>
<name>AwayTeamId</name>
</my:teamIdNames>
<xsl:variable name="vAttrNames" select=
"document('')/*/my:teamIdNames/*"/>
<xsl:template match="/">
<xsl:apply-templates select=
"//Fixture/@*[name()=$vAttrNames]"/>
</xsl:template>
<xsl:template match=
"@*[name()=document('')/*/my:teamIdNames/*]
[generate-id()
=
generate-id(key('kTeamById', .)[1])
]
">
<xsl:value-of select="."/><br />
</xsl:template>
</xsl:stylesheet>
在提供的XML文档上应用此转换时:
<League>
<Week Date="26/04/2010 19:00">
<Fixture Id="542" HomeTeamId="371"
HomeTeam="London Raiders Green"
AwayTeamId="13" AwayTeam="Richmond Swingers"/>
<Fixture Id="543" HomeTeamId="45"
HomeTeam="Spartans" AwayTeamId="15"
AwayTeam="Panthers"/>
<Fixture Id="544" HomeTeamId="370"
HomeTeam="Fat Cats" AwayTeamId="381"
AwayTeam="London Raiders Orange" />
</Week>
<Week Date="27/04/2010 19:00">
<Fixture Id="548" HomeTeamId="3"
HomeTeam="The Mob" AwayTeamId="81"
AwayTeam="London Raiders Red"/>
<Fixture Id="549" HomeTeamId="373"
HomeTeam="Intellect" AwayTeamId="83"
AwayTeam="Tornadoes"/>
</Week>
</League>
生成了想要的结果:
371<br/>13<br/>45<br/>15<br/>370<br/>381<br/>3<br/>81<br/>373<br/>83<br/>
请注意:
此解决方案不假设团队ID的来源仅来自名为HomeTeamId
和AwayTeamId
的属性。
事实上,可以使用任何一组变量名称(并包含在单独的文档中),因此这是一个非常强大的通用解决方案。