Как использовать ключи xslt 1.0 с вычисленными вместо статических значений узлов? - PullRequest
0 голосов
/ 14 февраля 2019

С помощью этого небольшого XSL-преобразования я могу придать «плоской» XML-структуре небольшую иерархию:

<?xml version="1.0"?>
<xsl:stylesheet xmlns:xsl="http://www.w3.org/1999/XSL/Transform" version="1.0">
    <xsl:output method="xml" version="1.0" encoding="UTF-8" indent="yes"/>
    <xsl:strip-space elements="*"/>
    <xsl:key name="child1" match="item[meta/para/level]" use="generate-id(preceding-sibling::item[meta/para/level &lt; current()/meta/para/level][1])"/>
    <xsl:key name="child2" match="item[meta/text]" use="generate-id(preceding-sibling::item[meta/para/level][1])"/>
    <xsl:template match="/doc">
        <xsl:variable name="min"> <!-- xslt2.0 min() -->
            <xsl:for-each select="item/meta/para/level">
                <xsl:sort select="." data-type="number"/>
                <xsl:if test="position() = 1">
                    <xsl:value-of select="."/>
                </xsl:if>
            </xsl:for-each>
        </xsl:variable>
        <root>
            <xsl:apply-templates select="item[meta/para/level = $min] | key('child2', '')"/>
        </root>
    </xsl:template>
    <xsl:template match="item[meta/para/level]">
        <test title="{meta/para/title}">
            <xsl:apply-templates select="key('child1', generate-id()) | key('child2', generate-id())"/>
        </test>
    </xsl:template>
    <xsl:template match="item[meta/text]">
        <text>
            <xsl:value-of select="meta/text"/>
        </text>
    </xsl:template>
</xsl:stylesheet>

Применительно к этому примеру XML с плоской структурой

<?xml version="1.0" encoding="UTF-8"?>
<doc>
    <item>
        <meta>
            <text>abc</text>
        </meta>
    </item>
    <item>
        <meta>
            <para>
                <level>1</level>
                <title>a</title>
            </para>
        </meta>
    </item>
    <item>
        <meta>
            <para>
                <level>2</level>
                <title>b</title>
            </para>
        </meta>
    </item>
    <item>
        <meta>
            <para>
                <level>3</level>
                <title>c</title>
            </para>
        </meta>
    </item>
    <item>
        <meta>
            <para>
                <level>4</level>
                <title>d</title>
            </para>
        </meta>
    </item>
    <item>
        <meta>
            <para>
                <level>4</level>
                <title>e</title>
            </para>
        </meta>
    </item>
    <item>
        <meta>
            <para>
                <level>5</level>
                <title>f</title>
            </para>
        </meta>
    </item>
    <item>
        <meta>
            <para>
                <level>2</level>
                <title>g</title>
            </para>
        </meta>
    </item>
    <item>
        <meta>
            <text>def</text>
        </meta>
    </item>
    <item>
        <meta>
            <text>ghi</text>
        </meta>
    </item>
    <item>
        <meta>
            <para>
                <level>4</level>
                <title>h</title>
            </para>
        </meta>
    </item>
</doc>

Является ли (желаемый) иерархический результат следующим образом:

<?xml version="1.0" encoding="UTF-8"?>
<root>
   <text>abc</text>
   <test title="a">
      <test title="b">
         <test title="c">
            <test title="d"/>
            <test title="e">
               <test title="f"/>
            </test>
         </test>
      </test>
      <test title="g">
         <text>def</text>
         <text>ghi</text>
         <test title="h"/>
      </test>
   </test>
</root>

Если информация об "уровне" определяется не целым числом, а длиной строки, преобразование не выполняется.

<?xml version="1.0" encoding="UTF-8"?>
<doc>
    <item>
        <meta>
            <text>abc</text>
        </meta>
    </item>
    <item>
        <meta>
            <para>
                <level>a</level>
                <title>a</title>
            </para>
        </meta>
    </item>
    <item>
        <meta>
            <para>
                <level>bc</level>
                <title>b</title>
            </para>
        </meta>
    </item>
    <item>
        <meta>
            <para>
                <level>def</level>
                <title>c</title>
            </para>
        </meta>
    </item>
    <item>
        <meta>
            <para>
                <level>ghij</level>
                <title>d</title>
            </para>
        </meta>
    </item>
    <item>
        <meta>
            <para>
                <level>klmn</level>
                <title>e</title>
            </para>
        </meta>
    </item>
    <item>
        <meta>
            <para>
                <level>opqrs</level>
                <title>f</title>
            </para>
        </meta>
    </item>
    <item>
        <meta>
            <para>
                <level>tu</level>
                <title>g</title>
            </para>
        </meta>
    </item>
    <item>
        <meta>
            <text>def</text>
        </meta>
    </item>
    <item>
        <meta>
            <text>ghi</text>
        </meta>
    </item>
    <item>
        <meta>
            <para>
                <level>vwxy</level>
                <title>h</title>
            </para>
        </meta>
    </item>
</doc>

Функция для нахождения минимума, который я расширил, вызывая «string-length»:

<xsl:sort select="string-length(.)" data-type="number"/>

Как использовать ключи так, чтобы длинаСтрока оценивается как информация уровня?

Ответы [ 2 ]

0 голосов
/ 17 февраля 2019

@ Вебби понял это почти правильно (спасибо!).При генерации ключа важно исключить неподходящие узлы: <xsl:key name="child1" match="item[meta/para]" use="generate-id(preceding-sibling::item[meta/para and (string-length(meta/para/level) &lt; string-length(current()/meta/para/level))][1])"/>, поэтому полный ответ будет

<?xml version="1.0"?>
<xsl:stylesheet xmlns:xsl="http://www.w3.org/1999/XSL/Transform" version="1.0">
    <xsl:output method="xml" version="1.0" encoding="UTF-8" indent="yes"/>
    <xsl:strip-space elements="*"/>
    <xsl:key name="child1" match="item[meta/para]"
use="generate-id(preceding-sibling::item[meta/para and (string-length(meta/para/level) &lt; string-length(current()/meta/para/level))][1])"/>
    <xsl:key name="child2" match="item[meta/text]"
use="generate-id(preceding-sibling::item[meta/para][1])"/>
    <xsl:template match="/doc">
        <xsl:variable name="min">
            <!-- xslt2.0 min() -->
            <xsl:for-each select="item/meta/para/level">
                <xsl:sort select="string-length(.)" data-type="number"/>
                <xsl:if test="position() = 1">
                    <xsl:value-of select="string-length(.)"/>
                </xsl:if>
            </xsl:for-each>
        </xsl:variable>
        <root>
            <xsl:apply-templates select="item[string-length(meta/para/level) = $min] | key('child2', '')"/>
        </root>
    </xsl:template>
    <xsl:template match="item[meta/para]">
        <test title="{meta/para/title}">
            <xsl:apply-templates select="key('child1', generate-id()) | key('child2', generate-id())"/>
        </test>
    </xsl:template>
    <xsl:template match="item[meta/text]">
        <text>
            <xsl:value-of select="meta/text"/>
        </text>
    </xsl:template>
</xsl:stylesheet>
0 голосов
/ 14 февраля 2019

В ключе child1, где вы сравниваете meta/para/level, вам нужно вычислить string-length для случая

, где <level> имеет строковое значение вместо числа .Как показано ниже:

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

<xsl:output method="xml" version="1.0" encoding="UTF-8" indent="yes"/>
<xsl:strip-space elements="*"/>

<xsl:key name="child1" match="item[meta/para/level]" 
use="generate-id(preceding-sibling::item[string-length(meta/para/level) &lt; string-length(current()/meta/para/level)][1])"/>

<xsl:key name="child2" match="item[meta/text]" 
use="generate-id(preceding-sibling::item[meta/para/level][1])"/>

<xsl:template match="/doc">
    <xsl:variable name="min"> <!-- xslt2.0 min() -->
        <xsl:for-each select="item/meta/para/level">

            <xsl:sort select="string-length(.)" data-type="number"/>
            <xsl:if test="position() = 1">
                <xsl:value-of select="."/>
            </xsl:if>
        </xsl:for-each>
    </xsl:variable>
    <root>
        <xsl:apply-templates select="item[meta/para/level = $min] | key('child2', '')"/>
    </root>
</xsl:template>
<xsl:template match="item[meta/para/level]">
    <test title="{meta/para/title}">
        <xsl:apply-templates select="key('child1', generate-id()) | key('child2', generate-id())"/>
    </test>
</xsl:template>
<xsl:template match="item[meta/text]">
    <text>
        <xsl:value-of select="meta/text"/>
    </text>
</xsl:template>
</xsl:stylesheet>

Пожалуйста, укажите тот же здесь

...