简体   繁体   English

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

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

I am trying to obtain the target xml such that value of a node is comma seperated values of sum of previous similar node values. 我试图获取目标xml,以使节点的值是先前相似节点值之和的逗号分隔值。 For eg: 例如:

Input: 输入:

<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>

Output: 输出:

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

As you can see above it is 50, (50+30), (50+30+70) 如上所示,它是50,(50 + 30),(50 + 30 + 70)

I tried with for-each Item and was able to find the sum of only current node and previous selected node. 我尝试使用for-each Item,并且能够找到仅当前节点和先前选择的节点的总和。 Can you please guide me here 你能在这里指导我吗

Try something like this: 尝试这样的事情:

<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>

While you could use sum(preceding-sibling::...) a more efficient solution would use a recursive template to accumulate the sum one value at a time: 虽然可以使用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>

If you want to solve it with XSLT 2.0 then I don't think you need any user defined function, you can write a single XPath 2.0 expression: 如果您想用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>

Thanks for the replies. 感谢您的答复。 I followed this approach which worked for me: 我遵循了对我有用的这种方法:

<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