使用变量中的其他节点(XSLT 1.0)在XSL apply-templates节点中排序

时间:2014-07-02 13:59:12

标签: xml sorting xslt xpath

如何使用$ bar中相应的节点顺序对$ foo中的<foo name="..."/>个节点进行排序?

部分解决方案(Mark Veenstra)

<xsl:for-each select="$list-resources">
    <xsl:apply-templates select="$building-resources[@name = current()/@name]">
        <xsl:with-param name="param" select="$some-value"/>
    </xsl:apply-templates>
</xsl:for-each>

新问题(请参阅问题末尾的编辑):

如何检索每个$ building-resources节点的整个排序节点集中的位置?


TL; DR回答

  • <xsl:sort/>有一个属性@data-type,用于指定@select节点中的数据类型。 默认为“text”。由于我比较数字,我必须设置 @data-type="number"。否则,在比较9和10时文本比较失败。

  • <xsl:sort select="count($bar[@name = current()/@name]/preceding-sibling::*)" data-type="number"/>

  • <xsl:sort/>中,current()指的是当前排序的节点(不是<xsl:template/>当前节点)

感谢michael.hor257k


完整答案

代码                       

    <!-- Resource "instances" to order, using $list-resources -->
    <xsl:variable name="building-resources"
        select="document('building.xml')/building/resource"/>

    <result>
        <!-- Apply templates to the resource instances -->
        <xsl:apply-templates select="$building-resources">
            <!-- Sort using the nodes order from the resources list -->
            <!-- Don't forget the @data-type: we're comparing NUMBERS not TEXTS -->
            <xsl:sort
                select="count($list-resources[@name = current()/@name]/preceding-sibling::*)"
                data-type="number"/>
        </xsl:apply-templates>
    </result>
</xsl:template>

<!-- Template to apply for each resource of $building-resources -->
<xsl:template match="ressource">
    <!-- position() refers to the node position in $building-resources
    because we used $building-resources as @select value
    of the apply-templates -->
    <output position="{position()}">
        <xsl:value-of select="."/>
    </output>
</xsl:template>

输入

列表resources.xml中

<resources>
    <resource name="wood"/>
    <resource name="stone"/>
    <resource name="gold"/>
</resources>

建筑物resources.xml中

<building>
    <resource name="stone"/>
    <resource name="gold"/>
    <resource name="stone"/>
    <resource name="wood"/>
    <resource name="wood"/>
</building>

输出

<result>
    <output position="1">wood</output>
    <output position="2">wood</output>
    <output position="3">stone</output>
    <output position="4">stone</output>
    <output position="5">gold</output>
</result>

原始问题

分拣

我有两个XML文档:

资源列表(list-resources.xml,如“类”列表):

<resources>
    <resource name="wood"/>
    <resource name="stone"/>
    <resource name="gold"/>
</resources>

建筑物内的资源(buildings.xml,如“对象”列表):

<building>
    <resource name="stone"/>
    <resource name="gold"/>
    <resource name="stone"/>
    <resource name="wood"/>
    <resource name="wood"/>
</building>

我想从<resource/>订购<building/>个节点,因此它与<resource/>的{​​{1}}订单相匹配。这是所需的输出:

<resources/>

为此,我在XSL中有两个节点集:

<result>
    <output position="1">wood</output>
    <output position="2">wood</output>
    <output position="3">stone</output>
    <output position="4">stone</output>
    <output position="5">gold</output>
</result>

我使用XSL apply-templates来处理来自<xsl:variable name="building-resources" select="building/resource"/> <xsl:variable name="list-resources" select="resources/resource"/> 的节点:

$building-resources

假设应用的模板没问题。我目前没有排序的结果是:

<xsl:apply-templates select="$building-resources">
    <xsl:with-param name="some-param" select="$variable-from-somewhere"/>
</xsl:apply-templates>

现在,我添加了一个<result> <output position="1">stone</output> <output position="2">gold</output> <output position="3">stone</output> <output position="4">wood</output> <output position="5">wood</output> </result> 元素来对<xsl:sort/>的节点进行排序,但我不知道要在$building-resources内添加什么...

@select

我尝试了以下XPath表达式:

<xsl:apply-templates select="$building-resources">
    <xsl:sort select="what-to-put-here"/>
    <xsl:with-param name="some-param" select="$variable-from-somewhere"/>
</xsl:apply-templates>

但在此XPath中,count($list-ressources[@name = current()/@name]/preceding-sibling::*) 指的是由current()处理的节点<xsl:template/>。 所以,而不是<xsl:apply-templates/> 但这不起作用,结果没有正确排序。 我的意思是current()进程当前正在处理的节点” (编辑:这实际上正是current()所做的!)在我的XPath括号内{ {1}}。我该怎么做?


获取排序位置

由于我简化了代码,我忘记了一些事情...... Mark Veenstra的解决方案引发了一个问题。

在应用的模板(生成<xsl:sort/>节点的模板)中,我使用$list-resources[]来获取应用模板的节点的位置:

<output/>

只要我只使用“one”apply-templates(因此应用模板中的position()保存节点在排序列表中的位置),这种方法很有效。但是,如果我应用Mark的解决方案(注意<xsl:template match="resource"> <xsl:variable name="position" select="position()"/> <output position="{$position}"> <xsl:value-of select="@name"/> </output> </xsl:template> 内的position()现在引用来自current()又来自$ liste-resources的节点:

<xsl:sort/>

然后,应用模板中的<xsl:for-each/>引用部分节点集中节点的位置($ building-resources [@name = current()/ @ name])

如何解决这个问题?

3 个答案:

答案 0 :(得分:2)

我认为你应该尝试做类似以下的事情。 (另)

变化:

<xsl:apply-templates select="$building-resources">
    <xsl:sort select="what-to-put-here"/>
    <xsl:with-param name="some-param" select="$variable-from-somewhere"/>
</xsl:apply-templates>

要:

<xsl:for-each select="$list-resources/@name">
    <xsl:variable name="curName" select="." />
    <xsl:apply-templates select="$building-resources[@name = $curName]">
        <xsl:with-param name="some-param" select="$variable-from-somewhere"/>
    </xsl:apply-templates>
</xsl:for-each>

因此,在应用模板之前,请确保按其他节点集的顺序应用。这就是为什么我在它周围添加了额外的for-each

答案 1 :(得分:2)

关于排序问题,我建议:

<xsl:stylesheet version="1.0" 
xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
<xsl:output method="xml" version="1.0" encoding="UTF-8" indent="yes"/>

<xsl:template match="/">
    <building>
        <xsl:apply-templates select="building/resource">
            <xsl:sort select="count(document('list-resources.xml')/resources/resource[@name = current()/@name]/preceding-sibling::resource)" data-type="number" order="ascending"/>
        </xsl:apply-templates>
    </building>
</xsl:template>

<xsl:template match="resource">
    <resource position="{position()}">
        <xsl:value-of select="@name"/>
    </resource>
</xsl:template>

</xsl:stylesheet>

应用于您的输入示例,返回:

<?xml version="1.0" encoding="UTF-8"?>
<building>
   <resource position="1">wood</resource>
   <resource position="2">wood</resource>
   <resource position="3">stone</resource>
   <resource position="4">stone</resource>
   <resource position="5">gold</resource>
</building>

我不确定其他问题是什么。

答案 2 :(得分:0)

我使用XSL密钥和<xsl:sort>,如下所示:

<xsl:key name="kResource" match="resources/resource" use="@name" />

<!-- regular identity template -->
<xsl:template match="node() | @*">
  <xsl:copy>
    <xsl:apply-templates select="node() | @*" />
  </xsl:copy>
</xsl:template>

<!-- specialized identity template, adds a position attribute -->
<xsl:template match="*" mode="with-position">
  <xsl:copy>
    <xsl:attribute name="position"><xsl:value-of select="position()" /></xsl:attribute>
    <xsl:apply-templates select="node() | @*" />
  </xsl:copy>
</xsl:template>

<xsl:template match="building">
  <result>
    <xsl:apply-templates select="*" mode="with-position">
      <xsl:sort select="count(key('kResource', @name)/preceding-sibling::*)" data-type="number" />
    </xsl:apply-templates>
  </result>
</xsl:template>

相关产出:

<result>
  <resource position="1" name="wood" />
  <resource position="2" name="wood" />
  <resource position="3" name="stone" />
  <resource position="4" name="stone" />
  <resource position="5" name="gold" />
</result>

请注意,即使我添加了position属性,我认为这是一件相当不必要的事情。 XML元素在文档中具有自然的位置,不需要过于具体。