在xslt 1.0中以逗号分隔值计算唯一值

时间:2015-06-23 07:46:48

标签: xslt xpath xslt-1.0

我在XML文件中有一个节点:

<TEST_STRING>12,13,12,14</TEST_STRING>

我需要计算这个字符串有多少个唯一数字/值。例如,在这种情况下,有2个唯一值,即13和14。

老实说,我还没有建造任何东西。在XSLT 1.0中似乎很难,但我的系统只支持1.0。

有什么解决方案吗?

2 个答案:

答案 0 :(得分:1)

如果您的处理器支持node-set扩展功能(exslt或Microsoft msxsl one),那么您可以分两步完成,首先拆分字符串并构建一个XML片段每个值一个元素,然后使用普通的XPath技术来查找单例。

第一步可以使用尾递归模板完成:

<xsl:template name="splitString">
  <xsl:param name="str"/>
  <xsl:choose>
    <xsl:when test="contains($str, ',')">
      <item><xsl:value-of select="substring-before($str, ',')"/></item>
      <xsl:call-template name="splitString">
        <xsl:with-param name="str" select="substring-after($str, ',')"/>
      </xsl:call-template>
    </xsl:when>
    <xsl:otherwise>
      <item><xsl:value-of select="$str"/></item>
    </xsl:otherwise>
  </xsl:choose>
</xsl:template>

在适当的地方,您可以将其称为

<xsl:variable name="itemsRTF">
  <xsl:call-template name="splitString">
    <xsl:with-param name="str" select="TEST_STRING"/>
  </xsl:call-template>
</xsl:variable>
<xsl:variable name="items" select="exsl:node-set($itemsRTF)"/>

items变量现在包含XML片段,如

<item>12</item>
<item>13</item>
<item>12</item>
<item>14</item>

接下来的挑战是找到单身,你可以用

这样的表达来做
$items/item[not(. = (preceding-sibling::item | following-sibling::item))]

(使用的方法更为有效,但对于少量项目,可能不值得烦恼)。所以要统计单身人士

<xsl:value-of select="count($items/item[not(. = (preceding-sibling::item | following-sibling::item))])"/>

答案 1 :(得分:-1)

没有节点集可能。当然,这不是最佳的(你好,Shlemiel The Painter),但它很容易 - 只有在前面或后面没有给定的字符串时才增加计数器。

<强>模板

<xsl:template name="calcUnique">
    <xsl:param name="str"/>
    <xsl:param name="back"/>
    <xsl:param name="count"/>

    <xsl:if test="$str">
        <xsl:choose>
            <xsl:when test="contains($str, ',')">
                <xsl:variable name="part" select="substring-before($str, ',')"/>
                <xsl:call-template name="calcUnique">
                    <xsl:with-param name="str" select="substring-after($str, ',')"/>
                    <xsl:with-param name="back" select="concat($back, ',', $part)"/>
                    <xsl:with-param name="count">
                        <xsl:choose>
                            <xsl:when test="contains(concat($str, ','), concat(',', $part, ',')) or contains(concat($back, ','), concat(',', $part, ','))">
                                <xsl:value-of select="$count"/>
                            </xsl:when>
                            <xsl:otherwise><xsl:value-of select="$count + 1"/></xsl:otherwise>
                        </xsl:choose>
                    </xsl:with-param>
                </xsl:call-template>
            </xsl:when>
            <xsl:otherwise>
                <xsl:choose>
                    <xsl:when test="contains(concat($back, ','), concat(',', $str, ','))">
                        <xsl:value-of select="$count"/>
                    </xsl:when>
                    <xsl:otherwise><xsl:value-of select="$count + 1"/></xsl:otherwise>
                </xsl:choose>
            </xsl:otherwise>
        </xsl:choose>
    </xsl:if>
</xsl:template>

示例用法

    <xsl:variable name="result">
        <xsl:call-template name="calc">
            <xsl:with-param name="str" select="TEST_STRING"/>
            <xsl:with-param name="back" select="''"/>
            <xsl:with-param name="count" select="0"/>
        </xsl:call-template>
    </xsl:variable>
    <xsl:value-of select="$result"/>