与此问题(http://stackoverflow.com/q/1333558/948404)类似,我想使用XPath计算这样结构中的产品总和:
<items>
<item>
<value>1.0</value>
<quantity>3</quantity>
</item>
<item>
<value>2.5</value>
<quantity>2</quantity>
</item>
<!-- ... -->
</items>
是否有XPath表达式计算每个项value
和quantity
的产品总和?
更新:解决方案必须使用PHP XSLTProcessor类,这意味着它可能必须符合XSLT 1.0。这就是为什么我还没有接受使用XSLT 2.0的两个可能正确的答案。我既不能在我的PHP实现中也不能在我的浏览器中测试它们,也不能在w3schools的Tryit Editor [1]中测试它们。遗憾!
答案 0 :(得分:4)
使用此XPath 2.0表达式:
sum(/items/item/(value * quantity))
以下是XSLT 2.0转换为验证:
<xsl:stylesheet version="2.0"
xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
<xsl:output method="text"/>
<xsl:template match="/">
<xsl:sequence select="sum(/items/item/(value * quantity))"/>
</xsl:template>
</xsl:stylesheet>
在提供的XML文档上应用此转换时:
<items>
<item>
<value>1.0</value>
<quantity>3</quantity>
</item>
<item>
<value>2.5</value>
<quantity>2</quantity>
</item>
<!-- ... -->
</items>
评估XPath表达式并输出此评估结果:
8
<强>解释强>:
在XPath 2.0中,位置步骤
是合法的 /(expression)
,
甚至
/someFunction(argumentList)
<强> II。 XSLT 1.0解决方案:
<xsl:stylesheet version="1.0"
xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
<xsl:output omit-xml-declaration="yes" indent="yes"/>
<xsl:template match="/*">
<xsl:call-template name="sumProducts">
<xsl:with-param name="pNodes" select="item"/>
</xsl:call-template>
</xsl:template>
<xsl:template name="sumProducts">
<xsl:param name="pNodes" select="/.."/>
<xsl:param name="pAccum" select="0"/>
<xsl:choose>
<xsl:when test="not($pNodes)">
<xsl:value-of select="$pAccum"/>
</xsl:when>
<xsl:otherwise>
<xsl:call-template name="sumProducts">
<xsl:with-param name="pNodes" select="$pNodes[position() >1]"/>
<xsl:with-param name="pAccum" select=
"$pAccum + $pNodes[1]/value * $pNodes[1]/quantity"/>
</xsl:call-template>
</xsl:otherwise>
</xsl:choose>
</xsl:template>
</xsl:stylesheet>
当在提供的XML文档(上面)上应用此转换时,再次生成所需的正确结果:
8
请注意:使用 FXSL library 可以轻松解决此类问题。要调用的模板是transform-and-sum
。
答案 1 :(得分:1)
试试这个:
<xsl:stylesheet version="2.0"xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
<xsl:output method="text"/>
<xsl:template match="/">
<xsl:sequence select="sum(/items/item/(value * quantity))"/>
</xsl:template>
</xsl:stylesheet>
答案 2 :(得分:1)
基于这个建议https://stackoverflow.com/a/1334165/948404我提出了一个符合XSLT 1.0和XPath 1.0的解决方案,它们由libxml / libxslt实现,由XSLTProcessor
的PHP实现使用(感谢@Dimitre Novatchev确认)。
<?xml version="1.0" encoding="UTF-8"?>
<xsl:transform version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
<xsl:template match="/items">
<xsl:call-template name="recursivesum">
<xsl:with-param name="items" select="*" />
</xsl:call-template>
</xsl:template>
<xsl:template name="recursivesum">
<xsl:param name="items" />
<xsl:param name="sum" select="0" />
<xsl:variable name="head" select="$items[1]" />
<xsl:variable name="tail" select="$items[position()>1]" />
<xsl:variable name="thissum" select="$head/value * $head/quantity" />
<xsl:choose>
<xsl:when test="not($tail)">
<xsl:value-of select="$sum+$thissum" />
</xsl:when>
<xsl:otherwise>
<xsl:call-template name="recursivesum">
<xsl:with-param name="sum" select="$sum+$thissum" />
<xsl:with-param name="items" select="$tail" />
</xsl:call-template>
</xsl:otherwise>
</xsl:choose>
</xsl:template>
</xsl:transform>