繁体   English   中英

XSLT中先前节点的总和作为当前节点的值

[英]sum of previous nodes as value of current node in XSLT

我试图获取目标xml,以使节点的值是先前相似节点值之和的逗号分隔值。 例如:

输入:

<Items>
  <Item>
    <name>Drakshi</name>
    <price>50</price>
  </Item>
  <Item>
    <name>Godambi</name>
    <price>30</price>
  </Item>
  <Item>
    <name>Badami</name>
    <price>70</price>
  </Item>
</Items>

输出:

<result>
 50,80,150
</result>

如上所示,它是50,(50 + 30),(50 + 30 + 70)

我尝试使用for-each Item,并且能够找到仅当前节点和先前选择的节点的总和。 你能在这里指导我吗

尝试这样的事情:

<xsl:template match="Items">
  <result>
    <xsl:for-each select="Item">
      <xsl:value-of select="sum(preceding-sibling::Item/price | price) " />
      <xsl:if test="position() != last()">, </xsl:if>
    </xsl:for-each>
  </result>
 </xsl:template>

虽然可以使用sum(preceding-sibling::...)但更有效的解决方案是使用递归模板来一次累加一个和值:

<xsl:template match="/Items">
    <result>
        <xsl:call-template name="run-total">
            <xsl:with-param name="values" select="Item/price"/>
        </xsl:call-template>
    </result>
</xsl:template>

<xsl:template name="run-total">
    <xsl:param name="values"/> 
    <xsl:param name="i" select="1"/> 
    <xsl:param name="total" select="0"/> 
    <xsl:variable name="balance" select="$total + $values[$i]" />
    <xsl:value-of select="$balance" />
    <xsl:if test="$i &lt; count($values)">
        <xsl:text>,</xsl:text>
        <xsl:call-template name="run-total">
            <xsl:with-param name="values" select="$values"/>
            <xsl:with-param name="i" select="$i + 1"/>
            <xsl:with-param name="total" select="$balance"/>
        </xsl:call-template>
    </xsl:if>
</xsl:template>

如果您想用XSLT 2.0解决它,那么我认为您不需要任何用户定义的函数,您可以编写一个XPath 2.0表达式:

<xsl:stylesheet xmlns:xsl="http://www.w3.org/1999/XSL/Transform"
    xmlns:xs="http://www.w3.org/2001/XMLSchema"
    exclude-result-prefixes="xs"
    version="2.0">

<xsl:template match="Items">
    <result>
        <xsl:value-of 
            select="for $pos in 1 to count(Item)
                    return sum(subsequence(Item, 1, $pos)/price)"
            separator=","/>
    </result>
</xsl:template>

</xsl:stylesheet>

感谢您的答复。 我遵循了对我有用的这种方法:

<xsl:function name="kk:getSumTillIndex">
    <xsl:param name="index"/>     
    <xsl:variable name="Var_PriceArray" select="$Items/item[position()&lt;=$index]//price/text()"/>
    <xsl:for-each select="$Var_PriceArray">
        <Item>
            <xsl:value-of select="current()"/>
        </Item>
    </xsl:for-each>
</xsl:function>


<xsl:function name="kk:calulatePriceSequence">
    <xsl:variable name="set" select="$Items/Item" />
    <xsl:variable name="count" select="count($set)" />

    <xsl:for-each select="$set">
         <xsl:if test="position() &lt;= $count">
         <xsl:value-of select="sum(kk:getSumTillIndex(position()))"/>

            <xsl:if test="position() &lt; number($count)-1">
                <xsl:value-of select="','" />
            </xsl:if>   
        </xsl:if>
        </xsl:for-each>
</xsl:function> 

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM