XSLT:循环并获取唯一的属性值

时间:2016-12-07 08:14:30

标签: xml for-loop foreach xslt-1.0

我必须遍历Main/MainLines/MainLine并得到总和(数量)for-each - Item/ItemNo。每个MainLine的输出应该只有一个ItemNo元素我正在做的事情,我正在重复。

输入

<Main Company="ABC" MainNo="213211" >
  <MainLines>
    <MainLine Quantity="2"  Node="9999">
      <Item ItemNo="123123" Class="NEW"/>
    </MainLine>
    <MainLine Quantity="1"  Node="9999">
      <Item ItemNo="123123" Class="NEW"/>
    </MainLine>
    <MainLine Quantity="3"  Node="9999">
      <Item ItemNo="123123" Class="NEW"/>
    </MainLine>
    <MainLine Quantity="2"  Node="9999">
      <Item ItemNo="22222" Class="NEW"/>
    </MainLine>
  </MainLines>
</Main>

输出

<Main Company="ABC" MainNo="213211" Reason="NewFile">
  <MainLines>
    <MainLine ExtnQuantity="6">
      <MainLine Quantity="2" Node="9999">
        <Item ItemNo="123123" Class="NEW" />
      </MainLine>
    </MainLine>
    <MainLine ExtnQuantity="6">
      <MainLine Quantity="1" Node="9999">
        <Item ItemNo="123123" Class="NEW" />
      </MainLine>
    </MainLine>
    <MainLine ExtnQuantity="6">
      <MainLine Quantity="3" Node="9999">
        <Item ItemNo="123123" Class="NEW" />
      </MainLine>
    </MainLine>
    <MainLine ExtnQuantity="2">
      <MainLine Quantity="2" Node="9999">
        <Item ItemNo="22222" Class="NEW" />
      </MainLine>
    </MainLine>
  </MainLines>
</Main>

XSL

<xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
  <xsl:output method="xml" indent="yes"/>
  <xsl:template match="/">
  <Main>
    <xsl:copy-of select="Main/@*"/>
    <xsl:attribute name="Reason">
      <xsl:value-of select="'NewFile'"/>
    </xsl:attribute>
    <MainLines>

      <xsl:for-each select="Main/MainLines/MainLine">
        <MainLine>
          <xsl:variable name="ITEM_ID">
            <xsl:value-of select="Item/@ItemNo"/>
          </xsl:variable>

          <xsl:attribute name="ExtnQuantity">
            <xsl:value-of select="sum(../MainLine[Item/@ItemNo = $ITEM_ID]/@Quantity)"/>
          </xsl:attribute>
          <xsl:copy-of select="."/>
        </MainLine>
      </xsl:for-each>

    </MainLines>
  </Main>
  </xsl:template>
</xsl:stylesheet>

预期产出

<Main Company="ABC" MainNo="213211" Reason="NewFile">
  <MainLines>
    <MainLine ExtnQuantity="6">
      <MainLine Quantity="2" Node="9999">
        <Item ItemNo="123123" Class="NEW" />
      </MainLine>
    </MainLine>
    <MainLine ExtnQuantity="2">
      <MainLine Quantity="2" Node="9999">
        <Item ItemNo="22222" Class="NEW" />
      </MainLine>
    </MainLine>
  </MainLines>
</Main>

1 个答案:

答案 0 :(得分:1)

这是一个Muenchian grouping问题。 您可以通过仅迭代ItemNo的唯一值来解决此问题。 以下是样式表的改编版本:

请注意,我已将<MainLines>元素移动到其自己的模板中以减少嵌套。此外,您可以直接向元素添加属性,您不需要<xsl:attribute>

<xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
  <xsl:output method="xml" indent="yes" />

  <!-- Define key -->
  <xsl:key name="ItemNoKey" match="MainLine/Item" use="@ItemNo" />

  <xsl:template match="/Main">
    <Main Reason="NewFile">
      <xsl:copy-of select="@*"/>
      <xsl:apply-templates select="MainLines"/>
    </Main>
  </xsl:template>

  <xsl:template match="MainLines">
    <MainLines>

      <!-- For each first MainLine with a given ItemNo -->
      <xsl:for-each select="MainLine[count(Item | key('ItemNoKey', Item/@ItemNo)[1]) = 1]">
        <xsl:variable name="CurrentItemNo" select="Item/@ItemNo"/>

        <!-- Print MainLine with quantity sum -->
        <MainLine ExtnQuantity="{sum(../MainLine[Item/@ItemNo = $CurrentItemNo]/@Quantity)}">

          <!-- Copy the selected MainLine element -->
          <xsl:copy-of select="."/>
        </MainLine>
      </xsl:for-each>

    </MainLines>
  </xsl:template>
</xsl:stylesheet>