XSLT 3中的条件流式累加器

时间:2018-02-26 07:22:02

标签: xslt-3.0

使用这样的输入XML:

<?xml version="1.0" encoding="UTF-8"?>
<Root>
  <Entry>
    <Amount>2088</Amount>
    <DebitCredit>C</DebitCredit>
  </Entry>
  <Entry>
    <Amount>9074</Amount>
    <DebitCredit>D</DebitCredit>
  </Entry>
  ...
</Root>

我想创建一个带有单独累加器的可流式转换,用于借记和放弃。信用,但是当我试图为一种类型或类似于此

的类型创建累加器时
<xsl:accumulator name="debitcount" initial-value="0" streamable="yes">
    <xsl:accumulator-rule phase="end" 
                          match="Entry[DebitCredit eq 'D']" 
                          select="$value + 1"/>  
</xsl:accumulator>

我发现显然匹配,选择或序列构造函数中的任何模式的扫描必须是静止的。我能够访问当前元素的属性值,但不能访问子元素或当前元素之前的任何内容。

我想知道我尝试做的事情是否可以在流媒体模式下使用累加器 - 我很确定我可以通过迭代器参数实现我的目标,但如果我正确理解文档,这似乎是一个非常大的限制。

1 个答案:

答案 0 :(得分:0)

根据我的理解,流式累加器存储元素内容的唯一方法是匹配文本子节点,例如。

<xsl:accumulator name="current-amount" as="xs:decimal?" initial-value="()" streamable="yes">
  <xsl:accumulator-rule match="Entry/Amount/text()" select="xs:decimal(.)"/>
</xsl:accumulator>

然后你应该能够在类似的规则中添加这些值,比如

<xsl:accumulator name="credit" as="xs:decimal" initial-value="0" streamable="yes">
  <xsl:accumulator-rule match="Entry/DebitCredit/text()" select="if (. = 'C') then $value + accumulator-before('current-amount') else $value"/>
</xsl:accumulator>

<xsl:accumulator name="debit" as="xs:decimal" initial-value="0" streamable="yes">
  <xsl:accumulator-rule match="Entry/DebitCredit/text()" select="if (. = 'D') then $value - accumulator-before('current-amount') else $value"/>
</xsl:accumulator>

如果您只想计算DebitCredit/text()[. = 'C']个节点的数量,那么应该采用相同的方法(即在文本子节点上进行匹配)。

如果你习惯普通的XSLT / XPath,我肯定会建议人们永远不要明确地选择文本节点,除非他们必须处理混合内容但是我发现使用流强制你改变你的XSLT / XPath编码方法相当多。

这是一个例子,输入为

<Root>
    <Entry>
        <Amount>100</Amount>
        <DebitCredit>C</DebitCredit>
    </Entry>
    <Entry>
        <Amount>50</Amount>
        <DebitCredit>D</DebitCredit>
    </Entry>
    <Entry>
        <Amount>10</Amount>
        <DebitCredit>C</DebitCredit>
    </Entry>
    <Entry>
        <Amount>20</Amount>
        <DebitCredit>D</DebitCredit>
    </Entry>
</Root>

Saxon 9.8.0.8 EE,带有样式表

<?xml version="1.0" encoding="UTF-8"?>
<xsl:stylesheet xmlns:xsl="http://www.w3.org/1999/XSL/Transform"
    xmlns:xs="http://www.w3.org/2001/XMLSchema"
    xmlns:math="http://www.w3.org/2005/xpath-functions/math"
    exclude-result-prefixes="xs math"
    version="3.0">

    <xsl:output method="text"/>

    <xsl:mode on-no-match="shallow-skip" streamable="yes" use-accumulators="#all"/>

    <xsl:accumulator name="current-amount" as="xs:decimal?" initial-value="()" streamable="yes">
        <xsl:accumulator-rule match="Entry/Amount/text()" select="xs:decimal(.)"/>
    </xsl:accumulator>

    <xsl:accumulator name="credit" as="xs:decimal" initial-value="0" streamable="yes">
        <xsl:accumulator-rule match="Entry/DebitCredit/text()" select="if (. = 'C') then $value + accumulator-before('current-amount') else $value"/>
    </xsl:accumulator>

    <xsl:accumulator name="debit" as="xs:decimal" initial-value="0" streamable="yes">
        <xsl:accumulator-rule match="Entry/DebitCredit/text()" select="if (. = 'D') then $value - accumulator-before('current-amount') else $value"/>
    </xsl:accumulator>

    <xsl:accumulator name="debitcount" initial-value="0" streamable="yes">
        <xsl:accumulator-rule 
            match="Entry/DebitCredit/text()[. = 'D']" 
            select="$value + 1"/>  
    </xsl:accumulator>

    <xsl:template match="/*" expand-text="yes">
        <xsl:apply-templates/>
        Debit count {accumulator-after('debitcount')}
        Sum of credits {accumulator-after('credit')},
        Sum of debits {accumulator-after('debit')}
    </xsl:template>

</xsl:stylesheet>

输出

   Debit count 2
   Sum of credits 110,
   Sum of debits -70