在一个或多个级别上进行XSLT 1.0分组

时间:2019-03-29 16:27:16

标签: xslt-1.0 xslt-grouping

今天的挑战是在XSLT 1.0中进行分组。发现有一些叫做键和Muenchian分组的东西。

输入XML:

<Items>
    <Item>
        <ID>1</ID>
        <Name>A</Name>
        <Country>Sweden</Country>
        <Region>Småland</Region>
    </Item>
    <Item>
        <ID>2</ID>
        <Name>B</Name>
        <Country>Sweden</Country>
        <Region>Norrland</Region>
    </Item>
    <Item>
        <ID>3</ID>
        <Name>C</Name>
        <Country>USA</Country>
        <Region>Alaska</Region>
    </Item>
    <Item>
        <ID>4</ID>
        <Name>D</Name>
        <Country>USA</Country>
        <Region>Texas</Region>
    </Item>
    <Item>
        <ID>5</ID>
        <Name>E</Name>
        <Country>Sweden</Country>
        <Region>Norrland</Region>
    </Item>
</Items>

我需要将XML精简为更好的结构,并且从此示例XML中我不希望按国家和地区来构造项目。以下是对国家和地区进行排序的结果:

<Items>
  <Country Name="Sweden">
    <Region Name="Norrland">
      <Item>
        <ID>2</ID>
        <Name>B</Name>
      </Item>
      <Item>
        <ID>5</ID>
        <Name>E</Name>
      </Item>
    </Region>
    <Region Name="Småland">
      <Item>
        <ID>1</ID>
        <Name>A</Name>
      </Item>
    </Region>
  </Country>
  <Country Name="USA">
    <Region Name="Alaska">
      <Item>
        <ID>3</ID>
        <Name>C</Name>
      </Item>
    </Region>
    <Region Name="Texas">
      <Item>
        <ID>4</ID>
        <Name>D</Name>
      </Item>
    </Region>
  </Country>
</Items>

编辑:

我还希望确保即使在重复的地区中也可以找到自己的国家。我相应地编辑了答案。

此外,我想向您暗示xsltfiddle.liberty-development.net是进行反复试验XSLT开发的一种简便方法...

1 个答案:

答案 0 :(得分:0)

article的启发,我找到了解决此问题的灵巧解决方案:

我已经包含了将其用于单组或双组的注释,请参见代码中的注释。请注意,我如何使用第一个键(索引)作为secon for-each循环的输入:

<?xml version="1.0" encoding="UTF-8"?>
<xsl:stylesheet version="1.0" 
    xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
  <xsl:output indent="yes"/>
  <xsl:strip-space elements="*"/>

  <xsl:key name="country" match="Item" use="Country" />
  <xsl:key name="region" match="Item" use="concat(Region, '|', Country)" />

  <xsl:template match="/Items">
    <Items>
      <xsl:for-each select="Item[generate-id(.) = generate-id(key('country', Country))]">
        <xsl:sort select="Country" />
        <xsl:variable name="_country" select="Country" />

        <xsl:element name="Country">
          <xsl:attribute name="Name"><xsl:value-of select="$_country" /></xsl:attribute>

          <!-- single level grouping -->
          <!--<xsl:apply-templates select="key('country', Country)" />-->


          <!-- double grouping -->
          <!-- START -->
          <xsl:for-each select="key('country', Country)[generate-id(.) = generate-id(key('region', concat(Region, '|', Country)))]">
              <xsl:sort select="Region" />
              <xsl:variable name="_region" select="Region" />

              <xsl:element name="Region">
                <xsl:attribute name="Name"><xsl:value-of select="$_region" /></xsl:attribute>

                <xsl:apply-templates select="key('region', concat(Region, '|', Country))" />
              </xsl:element>
          </xsl:for-each>
          <!-- END -->

        </xsl:element>    
      </xsl:for-each>
    </Items>
  </xsl:template>

  <xsl:template match="Item">
    <xsl:element name="Item">
      <xsl:element name="ID"><xsl:value-of select="ID" /></xsl:element>
      <xsl:element name="Name"><xsl:value-of select="Name" /></xsl:element>
    </xsl:element>
  </xsl:template>
</xsl:stylesheet>