Найти максимальное значение всех дочерних элементов и получить максимальную строку этого дочернего элемента, используя преобразование xsl 1.0 - PullRequest
1 голос
/ 29 апреля 2011

Например, исходный xml -

<records>
    <record>
        <personId>111</personId>
        <location>Australia</location>
        <year>1999</year>
    </record>
    <record>
        <personId>222</personId>
        <location>Netherland</location>
        <year>1919</year>
    </record>
    <record>
        <personId>111</personId>
        <location>USA</location>
        <year>2000</year>
    </record>
</records>

теперь после xsl это должно быть что-то вроде ниже, где оно берет последний год (максимум на один год) и отбрасывает старые записи для этого personID:

<records>
    <record>
        <personId>222</personId>
        <location>Netherland</location>
        <year>1919</year>
    </record>
    <record>
        <personId>111</personId>
        <location>USA</location>
        <year>2000</year>
    </record>
</records>

1 Ответ

2 голосов
/ 29 апреля 2011

Эта таблица стилей:

<xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
    <xsl:key name="kRecordByPersonId" match="record" use="personId"/>
    <xsl:template match="node()|@*" name="identity">
        <xsl:copy>
            <xsl:apply-templates select="node()|@*"/>
        </xsl:copy>
    </xsl:template>
    <xsl:template match="record"/>
    <xsl:template match="record[count(.|key('kRecordByPersonId',personId)[1])
                                 = 1]">
        <xsl:for-each select="key('kRecordByPersonId',personId)">
            <xsl:sort select="year" data-type="number" order="descending"/>
            <xsl:if test="position()=1">
                <xsl:call-template name="identity"/>
            </xsl:if>
        </xsl:for-each>
    </xsl:template>
</xsl:stylesheet>

Выход:

<records>
    <record>
        <personId>111</personId>
        <location>USA</location>
        <year>2000</year>
    </record>
    <record>
        <personId>222</personId>
        <location>Netherland</location>
        <year>1919</year>
    </record>
</records>

Просто для удовольствия, сохраняя порядок входного источника, но также и хорошую производительность:

<xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
    <xsl:key name="kRecordByPersonId" match="record" use="personId"/>
    <xsl:template match="node()|@*" name="identity">
        <xsl:copy>
            <xsl:apply-templates select="node()|@*"/>
        </xsl:copy>
    </xsl:template>
    <xsl:template match="records">
        <xsl:copy>
            <xsl:apply-templates select="@*"/>
            <xsl:call-template name="group-max">
                <xsl:with-param name="pGroup"
                 select="record[count(.|key('kRecordByPersonId',personId)[1])
                                 = 1]"/>
            </xsl:call-template>
        </xsl:copy>
    </xsl:template>
    <xsl:template name="group-max">
        <xsl:param name="pGroup" select="/.."/>
        <xsl:param name="pGroup-Max" select="/.."/>
        <xsl:choose>
            <xsl:when test="$pGroup">
                <xsl:for-each
                 select="key('kRecordByPersonId',$pGroup[1]/personId)">
                    <xsl:sort select="year"
                              data-type="number"
                              order="descending"/>
                    <xsl:if test="position()=1">
                        <xsl:call-template name="group-max">
                            <xsl:with-param name="pGroup"
                             select="$pGroup[position()!=1]"/>
                            <xsl:with-param name="pGroup-Max"
                             select="$pGroup-Max|."/>
                        </xsl:call-template>
                    </xsl:if>
                </xsl:for-each>
            </xsl:when>
            <xsl:otherwise>
                <xsl:apply-templates select="$pGroup-Max"/>
            </xsl:otherwise>
        </xsl:choose>
    </xsl:template>
</xsl:stylesheet>

Выход:

<records>
    <record>
        <personId>222</personId>
        <location>Netherland</location>
        <year>1919</year>
    </record>
    <record>
        <personId>111</personId>
        <location>USA</location>
        <year>2000</year>
    </record>
</records>
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...