首页 > 解决方案 > XSLT 2.0:通过匹配一个子节点并比较其他子节点来获取 XML 节点

问题描述

我输入了如下的 XML:

<parent>
    <payment>
        <id>123456</id>
        <type>CANCELLED</type>
        <date>2020-12-03</date>
        <amount>100</amount>
    </payment>
    <payment>
        <id>234567</id>
        <type>FORCE</type>
        <date>2020-12-01</date>
        <amount>200</amount>
    </payment>
    <payment>
        <id>345678</id>
        <type>CANCELLED</type>
        <date>2020-12-01</date>
        <amount>300</amount>
    </payment>
    <payment>
        <id>456789</id>
        <type>FORCE</type>
        <date>2020-12-01</date>
        <amount>400</amount>
    </payment>
    <payment>
        <id>456788</id>
        <type>CANCELLED</type>
        <date>2020-12-01</date>
        <amount>500</amount>
    </payment>
</parent>

现在,我们想要 type = 'CANCELLED' 的节点并且棘手的条件在这些取消的付款中,如果日期相等,那么我们需要选择具有最高 id 的节点。所以,在上面的示例输入中,应该只选择第 1 个和第 5 个节点。第一个节点的日期与任何其他节点都不匹配,因此被选中。虽然第 5 个节点的日期与第 3 个节点匹配,但第 5 个节点被选中,因为它具有更高的 id。

为了实现这一点,我尝试在 XSLT 下编写,

<xsl:stylesheet version="2.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
    <xsl:output method="xml" version="1.0" encoding="UTF-8" indent="yes"/>
    <xsl:strip-space elements="*"/>
    <xsl:variable name="input" select="parent"/>
    <xsl:template match="/">
        <xsl:for-each select="parent/payment[type='CANCELLED']">
            <xsl:choose>
                <xsl:when test="date = $input/payment[type='CANCELLED']/date">
                    <xsl:message>
                        <xsl:value-of select="amount"/>
                    </xsl:message>
                </xsl:when>
                <xsl:otherwise>
                    <xsl:value-of select="amount"/>
                </xsl:otherwise>
            </xsl:choose>
        </xsl:for-each>
    </xsl:template>
</xsl:stylesheet>

For-each 是必需的,因为在实际示例中我们正在提取一些值,因此我们需要调整 for-each 循环条件或在其中写入 if 条件。我尝试用整个输入 xml 声明变量输入,并比较在 for-each 迭代中选择的节点与输入节点,但随后它也与 self 匹配。我是否朝着正确的方向前进?请指导。

提前致谢。

标签: if-statementxsltforeachxslt-2.0

解决方案


要识别payment具有相同date使用分组的 s,然后在每个组中选择具有最高/最大 id 的项目:

<xsl:for-each-group select="//payment[type = 'CANCELLED']" group-by="date">
  <xsl:copy-of select="current-group()[id = max(current-group()/id)]"/>
</xsl:for-each-group>

推荐阅读