Группировка XSLT 1.0 на одном или нескольких уровнях - PullRequest
0 голосов
/ 29 марта 2019

Сегодняшняя проблема заключалась в группировке в XSLT 1.0. Обнаружено, что есть нечто, называемое ключами и мюнхенской группировкой.

Входной 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>

EDIT:

Я также хочу убедиться, что регионы окажутся в их собственной стране, даже если есть дубликаты. Я соответственно отредактировал ответ.

Кроме того, я хотел бы намекнуть о xsltfiddle.liberty-development.net как о простом способе разработки XSLT методом проб и ошибок ...

1 Ответ

0 голосов
/ 29 марта 2019

Вдохновленный этой статьей , я нашел изящное решение этой проблемы:

Я включил комментарии для его использования для одинарной или двойной группировки, см. Комментарии в коде.Обратите внимание, как я использую первый ключ (индекс) в качестве ввода для второго цикла 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>
...