Рекурсия: интегрирование сложного условия и порядка по рангу - PullRequest
0 голосов
/ 22 января 2020

Существующий XSLT-код необходимо усовершенствовать следующими условиями:

1 - Наряду с алгоритмом рекурсии добавить проверку того, что каждая сравниваемая строка (элемент) имеет:

  • статус атрибута = 0
  • , а атрибут status2 равен нулю (здесь это означает, что в элементе нет атрибута status2, фактически) (цепочка может быть создана только при выполнении этих двух условий)

2 - Порядок ранга. Наконец, все восстановленные (или восстанавливающие) цепочки должны быть упорядочены по атрибуту 'rank' (он может совпадать с нативной рекурсией, но иногда нет. В любом случае атрибут 'rank' имеет более высокий приоритет. Rank не имеет стабильной номенклатуры (например, 1-2- 3-4) но его цепочка может быть рассчитана путем сравнения «больше» - «меньше». (1 <3 <7). </p>

исходный код

<A>
  <X id="top"                         text="first"  text2="*"     status="0" rank="1"/>
  <X id="middle"  id-parent="top"     text="second" text2="**"    status="0" rank="3"/>
  <X id="bottom"  id-parent="middle"  text="third"  text2="***"   status="0" rank="6"/>

  <X id="bottom2" id-parent="middle"  text="fourth" text2="****"  status="0" rank="2"/> <!--note rank! bottom and middle should be switching because "bottom2 has a "higher" rank-->
  <X id="bottom3" id-parent="middle"  text="fifth"  text2="*****" status="2" rank="6" status2="any-value-make-its-status-not_null"/>
</A>

XSLT-преобразование

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

    <xsl:template match="@* | node()">
        <xsl:copy>
            <xsl:apply-templates select="@* | node()"/>
        </xsl:copy>
    </xsl:template>

    <xsl:key name="ref" match="X" use="@id"/>

    <xsl:template match="X">
        <xsl:copy>
            <xsl:copy-of select="@*"/>
            <xsl:variable name="Xs">
            <xsl:call-template name="XATT">
                <xsl:with-param name="currentX" select="."/>
                <xsl:with-param name="X" select=".">
                </xsl:with-param>
            </xsl:call-template>
            </xsl:variable>
            <xsl:if test="$Xs[node()]">
                <xsl:attribute name="chain-text">
                    <xsl:for-each select="$Xs/X">
                        <xsl:if test="position() ne 1"><xsl:text> | </xsl:text></xsl:if>
                        <xsl:value-of select="concat(@text, ' ', @text2)"/>
                    </xsl:for-each>
                </xsl:attribute>
                <xsl:for-each select="$Xs/X">
                    <xsl:attribute name="level-{position()}"><xsl:value-of select="concat(@text, ' | ', @text2)"/></xsl:attribute>
                </xsl:for-each>
            </xsl:if>

        </xsl:copy>
    </xsl:template>

    <xsl:template name="XATT">
        <xsl:param name="currentX"/>
        <xsl:param name="X"/>
        <xsl:choose>
            <xsl:when test="$currentX/@id-parent">
                <xsl:call-template name="XATT">
                    <xsl:with-param name="currentX" select="key('ref', $currentX/@id-parent)"/>
                    <xsl:with-param name="X">
                        <xsl:copy-of select="key('ref', $currentX/@id-parent)"/>
                        <xsl:copy-of select="$X"/>
                    </xsl:with-param>
                </xsl:call-template>
            </xsl:when>
            <xsl:otherwise>
                <xsl:copy-of select="$X"/>
            </xsl:otherwise>
        </xsl:choose>

    </xsl:template>

</xsl:stylesheet>

https://xsltfiddle.liberty-development.net/94AcskR

взаимосвязь схема

желаемый вывод

<?xml version="1.0" encoding="UTF-8"?><A>
  <X id="top" text="first" text2="*" chain-text="first *" level-1="first | *"/>
  <X id="middle" id-parent="top" text="second" text2="**" chain-text="first * | second **" level-1="first | *" level-2="second | **"/>
  <X id="bottom" id-parent="middle" text="third" text2="***" chain-text="first * | second ** | third ***" level-1="first | *" level-2="second | **" level-3="third | ***"/>

  <!--new conditions are clearly visible on the following lines-->
  <X id="bottom2" id-parent="middle"  text="fourth" text2="****" chain-text="first * | fourth **** | second **" level-1="first | *" level-2="fourth | ****" level-3="second | **"/> <!-- bottom and middle are switching because of 'rank' -->
  <X id="bottom3" id-parent="middle"  text="fifth"  text2="*****" status="1" rank="6" status2="any-value-here-make-its-status-not_null"/> <!--nothing happens with this line because of 'status' and 'status2' attributes  -->
</A>

1 Ответ

1 голос
/ 23 января 2020

Ваша формулировка скрыла проблему. Вам необходимо сложить иерархию (выраженную с помощью атрибутов @id и @id-parent) и затем отсортировать путь по атрибуту @rank.

Эта таблица стилей XSLT 1.0 (с расширениями EXSLT):

<xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform" 
    xmlns:exslt="http://exslt.org/common">
    <xsl:key name="X-By-id" match="X" use="@id"/>
    <xsl:key name="X-By-parent-id" match="X" use="string(@id-parent)"/>

    <xsl:variable name="fold-rtf">
        <xsl:apply-templates select="/" mode="fold"/>
    </xsl:variable>
    <xsl:variable name="folded-tree" select="exslt:node-set($fold-rtf)"/>

    <xsl:template match="@*|node()">
        <xsl:copy>
            <xsl:apply-templates select="@* | node()"/>
        </xsl:copy>
    </xsl:template>

    <xsl:template match="X[@status=0][not(@status2)]/@*[last()]">
        <xsl:variable name="current" select=".."/>
        <xsl:copy/>
        <xsl:for-each select="$folded-tree">
            <xsl:for-each select="key('X-By-id',$current/@id)">
                <xsl:attribute name="chain-text">
                    <xsl:for-each select="ancestor-or-self::*">
                        <xsl:sort select="@rank"/>
                        <xsl:if test="position()!=1"> | </xsl:if>
                        <xsl:value-of select="concat(@text,' ',@text2)"/>
                    </xsl:for-each>                
                </xsl:attribute>
                <xsl:for-each select="ancestor-or-self::*">
                    <xsl:sort select="@rank"/>
                    <xsl:attribute name="level-{position()}">
                        <xsl:value-of select="concat(@text,' | ',@text2)"/>
                    </xsl:attribute>
                </xsl:for-each>                
            </xsl:for-each>            
        </xsl:for-each>
    </xsl:template>

    <xsl:template match="/|*" mode="fold">
        <xsl:copy>
            <xsl:copy-of select="@*"/>
            <xsl:apply-templates select="key('X-By-parent-id',string(@id))" mode="fold">
            </xsl:apply-templates>
        </xsl:copy>
    </xsl:template>
</xsl:stylesheet>

Выходы:

<A>
  <X id="top" text="first" text2="*" status="0" rank="1" 
     chain-text="first *" level-1="first | *" />
  <X id="middle" id-parent="top" text="second" text2="**" status="0" rank="3" 
     chain-text="first * | second **" level-1="first | *" level-2="second | **" />
  <X id="bottom" id-parent="middle" text="third" text2="***" status="0" rank="6" 
     chain-text="first * | second ** | third ***" level-1="first | *" level-2="second | **" level-3="third | ***" />

  <X id="bottom2" id-parent="middle" text="fourth" text2="****" status="0" rank="2" 
     chain-text="first * | fourth **** | second **" level-1="first | *" level-2="fourth | ****" level-3="second | **" /> <!--note rank! bottom and middle should be switching because "bottom2 has a "higher" rank-->
  <X id="bottom3" id-parent="middle" text="fifth" text2="*****" status="2" rank="6" status2="any-value-make-its-status-not_null" />
</A>

Обратите внимание: использование функции node-set() для двухэтапного преобразования, изменение контекста для использования функции key() с другим входом документ, заставляющий приведение строки к ключу, потому что существуют специальные правила для наборов узлов в качестве ключей, а также потому, что я хотел использовать пустой ключ строки ...

Совет: все это может быть переведено в несколько строк XSLT 2.0 +

...