如何使用$ 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节点的整个排序节点集中的位置?
<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])
如何解决这个问题?
答案 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元素在文档中具有自然的位置,不需要过于具体。