1

我有一些来自 3rd 方软件的输出:

来自软件的示例 XML:

<testsuite name="XYZ">  
    <testcase name="ABC" status="0" time="12.001">Some stuff</testcase>  
    <testcase name="DEF" status="0" time="12,345.001">Some stuff</testcase>  
    <testcase name="GHI" status="0" time="4,321.001">Some stuff</testcase>     
</testsuite>

我需要编写一个 XSLT 将其转换为以下内容:

<testsuite name="XYZ" time="16678.003">
    <testcase name="ABC" time="12.001">Some stuff</testcase>
    <testcase name="DEF" time="12,345.001">Some stuff</testcase>
    <testcase name="GHI" time="4,321.001">Some stuff</testcase>    
</testsuite>

我几乎到了那里,除了testsuite元素的time属性。我没有得到总数,而是得到NaN。我用来得到这个的 XPath 表达式是sum(//testsuite/@time)

请注意,当所有时间都 < 1000 时不会发生错误。这可能是因为 XSLT 在遇到逗号时不解析数字。(我无法从输入中删除这些逗号,因为它来自第 3 方软件。)

那么我如何对这些时间值求和呢?是否可以修改sum(//testsuite/@time)以便能够即时删除逗号?

谢谢!

4

2 回答 2

3

像这样使用 XSLT 的translate函数

sum(translate(//testsuite/@time,',',''))

于 2009-10-28T11:12:12.597 回答
1

使用此递归模板将所有时间属性添加在一起:

<xsl:template name="sumoftime">
  <xsl:param name="node"/>
  <xsl:param name="sum" />
  <xsl:choose>
    <xsl:when test="$node">
      <xsl:call-template name="sumoftime">
        <xsl:with-param name="node" select="$node/following-sibling::testcase[1]"/>
        <xsl:with-param name="sum" select="$sum + number(translate($node/@time, ',', ''))"/>
      </xsl:call-template>
    </xsl:when>
    <xsl:otherwise>
      <xsl:value-of select="$sum"/>
    </xsl:otherwise>
  </xsl:choose>
</xsl:template>

然后在你的 testsuite 元素中,调用上面的模板

<xsl:element name="testsuite">

            ...

    <xsl:attribute name="{'time'}">
        <xsl:call-template name="sumoftime">
            <xsl:with-param name="node" select="testcase[1]"/>
            <xsl:with-param name="sum" select="0" />
        </xsl:call-template>
    </xsl:attribute>
</xsl:element>

相当复杂,但如果您仅限于 XSLT 1.0,我认为这是可行的方法。

模板的功劳应该归功于 Jelovirt,因为这只是他对另一个问题的回答的一个小改动:stackoverflow.com/questions/647991/summing-numbers-with-comma-as-decimal-separator-in-xslt

于 2009-10-28T22:56:54.893 回答