如何将xpath表达式转换为xsl键?

时间:2013-04-19 16:51:40

标签: xslt xpath xslt-1.0

我坚持使用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

我做错了什么?

1 个答案:

答案 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="'&#xA;'" />

  <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