1

我有一个样本”:

<Root>
    <A rename="yes,it is option 1"/>
    <C rename="no"/>
    <A rename="yes,it is option 2"/>
    <C rename="no"/>
    <C rename="yes"/>
    <C rename="no"/>  
    <A rename="yes,it is option 3"/>
    <A rename="yes,it is option 4"/>
    <C rename="no"/>
    <C rename="yes"/>
    <C rename="no"/>      
    <C rename="no"/>        
</Root>

然后我应用一个如下所示的模板:

   <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="A">
            <p><b>option1: <xsl:value-of select="count(following-sibling::C[preceding-sibling::A[1]/@rename[contains(.,'option 1')]])"/></b></p>
            <p><b>option2: <xsl:value-of select="count(following-sibling::C[preceding-sibling::A[1]/@rename[contains(.,'option 2')]])"/></b></p>            
            <p><b>option3: <xsl:value-of select="count(following-sibling::C[preceding-sibling::A[1]/@rename[contains(.,'option 3')]])"/></b></p>
            <p><b>option4: <xsl:value-of select="count(following-sibling::C[preceding-sibling::A[1]/@rename[contains(.,'option 4')]])"/></b></p>                        
 </xsl:template>
</xsl:stylesheet>

但我希望输出看起来像这样,如果 A 后面没有兄弟,我们就忽略这个。只需打印这些 @rename 包含“选项”并且还包含元素

option1: 1 

option2: 3 

option4: 4 

我现在得到的是:

option1: 1 

option2: 3 

option3: 0 

option4: 4 

option1: 0 

option2: 3 

option3: 0 

option4: 4 

option1: 0 

option2: 0 

option3: 0 

option4: 4 

option1: 0 

option2: 0 

option3: 0 

option4: 4 
4

2 回答 2

1

您正在计算最近的 A元素之后的所有C元素。这可以通过一个键来完成,将所有此类C元素与关联的 A元素分组

<xsl:key name="lookup" match="C" use="generate-id(preceding-sibling::A[1])" />

要查找至少存在以下C的 A元素,请执行以下操作:

<xsl:apply-templates select="A[key('lookup', generate-id())]" />

然后要计算选项的数量,您可以只计算键中的元素数量,如下所示:

<xsl:value-of select="count(key('lookup', generate-id()))" />

尝试以下 XSLT

<xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
    <xsl:output method="text" indent="yes"/>
    <xsl:key name="lookup" match="C" use="generate-id(preceding-sibling::A[1])" />

    <xsl:template match="/Root">
        <xsl:apply-templates select="A[key('lookup', generate-id())]" />
    </xsl:template>

    <xsl:template match="A">
        <xsl:value-of select="concat(@rename, ': ', count(key('lookup', generate-id())) , '&#13;&#13;')" />
    </xsl:template>
</xsl:stylesheet> 

当这应用于您的 XML 示例时,将输出以下内容:

yes,it is option 1: 1

yes,it is option 2: 3

yes,it is option 4: 4

请注意,如果您想要“option1”而不是“是的,它是选项 1”,您可以这样做concat('option', substring-after(@rename, 'option '))而不是仅仅@rename

请注意,如果您重新构建 XML,则可以简化此问题。这样的事情会好得多:

<Root>
    <A rename="yes,it is option 1">
        <C rename="no"/>
    </A>
    <A rename="yes,it is option 2">
        <C rename="no"/>
        <C rename="yes"/>
        <C rename="no"/>
    </A>
    <A rename="yes,it is option 3"/>
    <A rename="yes,it is option 4">
        <C rename="no"/>
        <C rename="yes"/>
        <C rename="no"/>
        <C rename="no"/>
    </A>
</Root>
于 2012-05-01T07:50:11.797 回答
0

在 XSLT 2.0 中,这变成了一个简单的分组问题,使用xsl:for-each-group指令。

<xsl:template match="Root">
  <xsl:for-each-group group-starting-with="A" select="*">
    <xsl:if test="count(current-group()) != 1">
      <p>Option <xsl:value-of select="substring-after(current-group()[1]/@rename, 'option ')"/> = <xsl:value-of select="count(current-group())-1"/></p>        </xsl:if>
  </xsl:for-each-group>
</xsl:template>
于 2012-05-01T08:05:53.413 回答