我坚持使用XSLT密钥应该是一个简单的问题。
如果它是相关的,我被迫进入XSLT 1.0解析器。
示例XML:
<updates>
<update>
<id>first</id>
<pkglist>
<collection arch='i686'>
<package name='bin' version='1.0'/>
</collection>
<collection arch='x86_64'>
<package name='bin' version='1.0'/>
</collection>
</pkglist>
</update>
<update>
<id>second</id>
<pkglist>
<collection arch='i686'>
<package name='bin' version='1.1'/>
<package name='conf' version='1.1'/>
</collection>
<collection arch='x86_64'>
<package name='bin' version='1.2'/>
<package name='conf' version='1.1'/>
</collection>
</pkglist>
</update>
<update>
<id>third</id>
<pkglist>
<collection arch='i686'>
<package name='bin' version='1.3'/>
<package name='conf' version='1.1'/>
<package name='src' version='1.3'/>
</collection>
<collection arch='x86_64'>
<package name='bin' version='1.3'/>
<package name='conf' version='1.2'/>
<package name='src' version='1.3'/>
</collection>
</pkglist>
</update>
</updates>
这个XPATH选择我正在寻找的东西
/updates//update[pkglist/collection/package/@name = 'bin']/id/text()
我正在查找整个文档中包含属性为“name”的包的所有“id”值。但在现实世界中,我有更多的包装,而不是手工列出的明智之处。
所以我认为关键是要走的路
<xsl:key name="idByPackage" match="/updates//update/pkglist/collection/package/@name" use="../../../../id" />
但这并没有给我任何有用的东西
<xsl:key name="idByPackage" match="/updates//update/pkglist/collection/package/@name" use="../../../../id" />
<xsl:template match="/">
<xsl:apply-templates select="updates/update" />
</xsl:template>
<xsl:template match="update">
Updates:
<xsl:value-of select="id" />
Related updates
<xsl:apply-templates select="pkglist" />
</xsl:template>
<xsl:template match="pkglist">
<xsl:for-each select="key('idByPackage', collection/package/@name)">
<xsl:value-of select="." />
<xsl:value-of select="collection/package/@name" />
</xsl:for-each>
</xsl:template>
我知道我在正确的区域,因为当我改变钥匙时:
<xsl:key name="idByPackage" match="/updates//update/pkglist/collection/package/@name" use="../../../../pkglist/collection/package/@name" />
相同的xsl模板spits打包我的包名。
当我使用'看起来对我有效但不起作用'键运行模板时,我得到了这个:
Updates:
first
Related updates
Updates:
second
Related updates
Updates: third
Related updates
当我希望得到像
这样的东西时Updates:
first
Related updates
second
third
Updates:
second
Related updates
first
third
Updates: third
Related updates
first
second
我做错了什么?
答案 0 :(得分:3)
这是您需要的钥匙:
<xsl:key name="idByPackage"
match="update/id" use="../pkglist/collection/package/@name" />
在您的示例输入上运行此XSLT时:
<xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
<xsl:output method="text" indent="yes" omit-xml-declaration="yes"/>
<xsl:key name="idByPackage"
match="update/id" use="../pkglist/collection/package/@name" />
<xsl:variable name="nl" select="'
'" />
<xsl:template match="/">
<xsl:apply-templates select="updates/update" />
</xsl:template>
<xsl:template match="update">
<xsl:value-of select="concat('Updates:', $nl,
id, $nl,
'Related updates:', $nl)" />
<xsl:variable name="name" select="pkglist/collection/package/@name" />
<xsl:apply-templates select="key('idByPackage', $name)[. != current()/id]" />
</xsl:template>
<xsl:template match="id">
<xsl:value-of select="concat(., $nl)" />
</xsl:template>
</xsl:stylesheet>
结果是:
Updates:
first
Related updates:
second
third
Updates:
second
Related updates:
first
third
Updates:
third
Related updates:
first
second