好的,我有这个xml
<roots>
<root>
<name>first</name>
<item type='test'><something>A</something></item>
<item type='test'><something>B</something></item>
<item type='test'><something>C</something></item>
<item type='test'><something>A</something></item>
<item type='other'><something>A</something></item>
<item type='test'><something>B</something></item>
<item type='other'><something>D</something></item>
</root>
<root>
<name>second</name>
<item type='test'><something>E</something></item>
<item type='test'><something>B</something></item>
<item type='test'><something>F</something></item>
<item type='test'><something>A</something></item>
<item type='other'><something>A</something></item>
<item type='test'><something>B</something></item>
<item type='other'><something>D</something></item>
</root>
</roots>
现在我需要获取每个根节点的唯一值到目前为止我有
<?xml version="1.0" encoding="ISO-8859-1"?>
<xsl:stylesheet version="1.0"
xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
<xsl:output indent="yes" method="text"/>
<xsl:key name="item-by-value" match="something" use="."/>
<xsl:key name="rootkey" match="root" use="name"/>
<xsl:template match="/">
<xsl:for-each select="key('rootkey','second')">
<xsl:for-each select="item/something">
<xsl:if test="generate-id() = generate-id(key('item-by-value', normalize-space(.)))">
<xsl:value-of select="."/>
</xsl:if>
</xsl:for-each>
</xsl:for-each>
</xsl:template>
</xsl:stylesheet>
如果我使用“First”作为获得第一根的关键我得到一个好的结果ABCD
如果我使用“秒”,我只获得EF,但我需要结果为ABDFE
答案 0 :(得分:2)
这是您必须使用复合键的情况。
此转化:
<xsl:stylesheet version="1.0"
xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
<xsl:output method="text"/>
<xsl:key name="kSomethingByNameAndVal" match="something"
use="concat(../../name, '+', .)"/>
<xsl:template match="/">
<xsl:for-each select="*/*">
<xsl:for-each select=
"item/something
[generate-id()
=
generate-id(key('kSomethingByNameAndVal',
concat(../../name, '+', .)
)
)
]
">
<xsl:value-of select="."/>
</xsl:for-each>
<xsl:text>
</xsl:text>
</xsl:for-each>
</xsl:template>
</xsl:stylesheet>
在应用于提供的XML文档时正确生成所需结果:
<强> ABCD
强>
<强> EBFAD
强>
答案 1 :(得分:2)
我稍微修改了你的xsl,我得到了EBFAD
。关键是如果您使用密钥在给定的根元素下找到具有此内容的第一个节点,则该密钥需要特定于根元素。我将xsl:key
更改为:
<xsl:key name="item-by-value" match="something"
use="concat(normalize-space(.), ' ', generate-id(./ancestor::root))"/>
然后xsl:if
测试成为:
<xsl:if test="generate-id() = generate-id(key('item-by-value',
concat(normalize-space(.), ' ', generate-id(./ancestor::root))))">
答案 2 :(得分:0)
我觉得你很亲密。但是,您只需要具有正确root作为父项的item-by-value键的第一个值:
<xsl:for-each select="key('rootkey','second')">
<xsl:variable name="root" select="generate-id()" />
<xsl:for-each select="item/something">
<xsl:sort order="ascending" select="." data-type="text"/>
<xsl:if test="generate-id() = generate-id(key('item-by-value', .)[generate-id(ancestor::root)=$root][1])">
<xsl:value-of select="."/>
</xsl:if>
</xsl:for-each>
</xsl:for-each>