在XSLT中按父节点分组

时间:2014-02-28 11:27:37

标签: xml xslt xslt-1.0

问题的标题可能令人困惑。如果有人有正确的术语,请编辑。 我有一个像这样的XML输入文件:

    <?xml version="1.0"?>
    <?xml-stylesheet type="text/xsl" href="show.xsl"?>
    <patients>
        <patient>
            <name>Joe</name>
            <age>45</age>
            <kids>
                <kid>
                    <name>Ramsay</name>
                    <gender>M</gender>
                </kid>
                <kid>
                    <name>Eva</name>
                    <gender>F</gender>
                </kid>
                <kid>
                    <name>Arthas</name>
                    <gender>M</gender>
                </kid>
            </kids>
        </patient>
        <patient>
            <name>Adam</name>
            <age>34</age>
            <kids>
                <kid>
                    <name>Jon</name>
                    <gender>M</gender>
                </kid>
                <kid>
                    <name>Jane</name>
                    <gender>F</gender>
                </kid>
                <kid>
                    <name>Sunita</name>
                    <gender>F</gender>
                </kid>
            </kids>
        </patient>
    </patients>

我希望获得类似于此的输出:

    Name: Joe
    Age: 45
    Kids:
        Boys:
            Ramsay
            Arthas
        Girls:
            Eva
    Name: Adam
    Age: 34
    Kids:
        Boys:
            Jon
        Girls:
            Jane
            Sunita

这是我使用xsl:key编写的XSLT,但它提供了错误的输出,因为它无法按患者分组。     

<xsl:stylesheet version="1.0"
xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
<xsl:key name="header" match="//kid" use="gender"/>
<xsl:template match="//patients/patient">
        <xsl:value-of select="name"/>
        <xsl:text xml:space="preserve">
        </xsl:text>
        <xsl:value-of select="age"/>
        <xsl:text xml:space="preserve">
        </xsl:text>
        <xsl:text xml:space="preserve">Boys:
            </xsl:text>
        <xsl:for-each select="key('header', 'M')">
            <xsl:value-of select="name"/>
            <xsl:text xml:space="preserve">
            </xsl:text>
        </xsl:for-each>
        <xsl:text xml:space="preserve">
        </xsl:text>        
<xsl:text xml:space="preserve">Girls:
            </xsl:text>
        <xsl:for-each select="key('header', 'F')">
            <xsl:value-of select="name"/>
            <xsl:text xml:space="preserve">
            </xsl:text>
        </xsl:for-each>
</xsl:template>

</xsl:stylesheet>

我正在寻找一种解决方案,最好是在XSLT 1.0中。

1 个答案:

答案 0 :(得分:0)

如果您的数据没有唯一的患者ID(或者即使它有),您可以使用generate-id()函数生成一个。我不确定你是否有一个(通常,一个名称不能被认为是唯一的),所以试试这样:

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

<xsl:output method="text" encoding="UTF-8" />
<xsl:strip-space elements="*"/>

<xsl:key name="kids-by-gender" match="kid" use="concat(generate-id(ancestor::patient), gender)"/>

<xsl:template match="patient">
    <xsl:variable name="pid" select="generate-id()" />
    <xsl:value-of select="concat('Name: ', name, '&#10;') "/>
    <xsl:value-of select="concat('Age: ', age, '&#10;') "/>
    <xsl:text>Kids:&#10;</xsl:text>
    <xsl:text>&#9;Boys:&#10;</xsl:text>
    <xsl:apply-templates select="key('kids-by-gender', concat($pid, 'M')) "/>
    <xsl:text>&#9;Girls:&#10;</xsl:text>
    <xsl:apply-templates select="key('kids-by-gender', concat($pid, 'F')) "/>
    <xsl:text>&#10;</xsl:text>
</xsl:template>

<xsl:template match="kid">
    <xsl:value-of select="concat('&#9;&#9;', name, '&#10;') "/>
</xsl:template>
</xsl:stylesheet>

请注意,这需要更多的工作来容纳没有性别的孩子或根本没有孩子的患者。