使用XSLT基于属性合并Xml元素

时间:2015-01-08 00:06:08

标签: xml xslt

我想合并位置元素,并在比较type属性后创建一个名为travel的新元素。第一个类型=“从”必须合并第一个类型=“到”找到,并应继续其余。

INPUT

<?xml version="1.0" encoding="UTF-8"?>
<JOB age="0" priority="N">
    <container ID="TSTU2345678" TP="4200"/>
    <container ID="TSTU3456789" TP="4200"/>
    <position refID="Y.Test:AA.01.01.1" name="AA0101.1" type="from"/>
    <position refID="Y.Test:AA.01.02.1" name="AA0102.1" type="from"/>
    <position refID="Y.Test:AA.02.02.1" name="AA0202.1" type="to"/>
    <position refID="Y.Test:AA.02.03.1" name="AA0203.1" type="to"/>
</JOB>

必需的输出

<JOB age="0" priority="N">
    <Travel FromrefID="Y.Test:AA.01.01.1" Fromname="AA0101.1" TorefID="Y.Test:AA.02.02.1" Toname="AA0202.1"/>
    <Travel FromrefID="Y.Test:AA.01.02.1" Fromname="AA0102.1" TorefID="Y.Test:AA.02.03.1" Toname="AA0203.1"/>
</JOB>

到目前为止我尝试了什么。它将所有To类型添加到元素中。我不知道如何选择第一个并将其标记为已使用。另外,我如何更改属性名称。请帮忙。

<xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
    <xsl:output method="xml" indent="yes"/>
    <xsl:key name="position" match="job/position" use="@type" />
    <xsl:template match="job">
        <xsl:copy>
            <xsl:apply-templates select="@*" />
        </xsl:copy>
    </xsl:template>
    <xsl:template match="/">
        <JOB>
            <xsl:apply-templates select="job/@*" />
            <xsl:for-each select="key('position','from')">
                <Travel>
                    <xsl:apply-templates select="@*" />

                    <xsl:for-each select="key('position','to')">
                        <To>
                            <xsl:apply-templates select="@*" />
                        </To>
                    </xsl:for-each>
                </Travel>

            </xsl:for-each>
        </JOB>
    </xsl:template>

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

</xsl:stylesheet>

感谢。

2 个答案:

答案 0 :(得分:3)

我建议你这样看:

XSLT 1.0

<xsl:stylesheet version="1.0" 
xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
<xsl:output method="xml" version="1.0" encoding="utf-8" indent="yes"/>

<xsl:key name="to" match="position[@type='to']" use="count(preceding-sibling::position[@type='to'])" />

<xsl:template match="/JOB">
    <xsl:copy>
        <xsl:copy-of select="@*"/>
        <xsl:for-each select="position[@type='from']">
            <xsl:variable name="to" select="key('to', position() - 1)" />
            <Travel FromrefID="{@refID}" Fromname="{@name}" TorefID="{$to/@refID}" Toname="{$to/@name}"/>
        </xsl:for-each>
    </xsl:copy>
</xsl:template>

</xsl:stylesheet>

请注意,我们在此假设有相同数量的&#34;来自&#34;和&#34;到&#34;职位(或至少&#34;来自&#34;职位的数量不小于&#34;至&#34;职位的数量。)

答案 1 :(得分:1)

替代XSLT 1.0解决方案

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

    <xsl:template match="@*|node()">
        <xsl:copy>
            <xsl:apply-templates select="@*|position[@type='from']"/>
        </xsl:copy>
    </xsl:template>

    <xsl:template match="position[@type='from']">
        <xsl:variable name="fromPos" select="position()"/>
        <Travel FromRefId="{@refID}" Fromname="{@name}">
            <xsl:apply-templates select="position[@type='to'][position()=$fromPos]"/>
        </Travel>
    </xsl:template>

    <xsl:template match="position[@type='to']">
        <xsl:attribute name="TorefID">
            <xsl:value-of select="@refID"/>
        </xsl:attribute>
        <xsl:attribute name="Toname">
            <xsl:value-of select="@name"/>
        </xsl:attribute>
    </xsl:template>

</xsl:stylesheet>