如何从一个XML元素组中获取数据并将其提供给另一个XML组

时间:2015-01-08 01:56:46

标签: xslt xslt-1.0 xslt-2.0 xslt-grouping exslt

我正在尝试使用辅助或不同xml文件中的数据更新以下XML中的元素。输入XML和辅助XML文件都具有相同数量的段。我需要在辅助XML的第一段中获取值并更新INPUT xml中的元素,依此类推。我不确定是否可以使用XSL完成,任何人都可以指导我。

更具体地说,我正在尝试根据辅助XML // PDetails / PStatus / Code和// PDetails / PStatus / Description值更新每个<indicator></indicator>中的INPUT XML <iOSection>值。 / p>

以下是INPUT XML文件:

<IResponse>
    <iOSection>
        <Details>           
            <Info>
                <pNumber>FB061689</pNumber>
                <indicator></indicator>
                <Identifier>1</Identifier>
            </Info>         
        </Details>
        <Token>
            <Reference>1UUYD05BHM21OJCK3881C7F</Reference>
        </Token>
    </iOSection>
    <iOSection>
        <Details>           
            <Info>
                <pNumber>FB061690</pNumber>
                <indicator></indicator>
                <Identifier>2</Identifier>
            </Info>
        </Details>
        <Token>
            <Reference>1UUYD05BHM21OJCK3881C7F</Reference>
        </Token>
    </iOSection>
</IResponse>

下面是辅助XML文件 - 它在名为RSPDetails的xsl变量中可用

<RS PartID="abcd" SysID="mnc">  
    <PDetails>
        <PN>FB063586</PN>
        <PStatus>
            <Code>0</Code>
            <Description>Cancelled</Description>
        </PStatus>     
    </PDetails>
    <PDetails>
        <Error>
            <Code>92</Code>
            <Message>failed</Message>
        </Error>
    </PDetails>
</RS>

当// PDetails / PStatus / Code ='0'和// PDetails / PStatus / Description ='已取消'时,<indicator>的值应为'YES',在所有其他情况下,它应为'NO “

条件应适用于使用<iOSection>位置1数据的<PDetails>位置1和使用<iOSection>位置2数据的<PDetails>位置2等等

期望OUTPUT是:

<IResponse>
    <iOSection>
        <Details>           
            <Info>
                <pNumber>FB061689</pNumber>
                <indicator>YES</indicator>
                <Identifier>1</Identifier>
            </Info>         
        </Details>
        <Token>
            <Reference>1UUYD05BHM21OJCK3881C7F</Reference>
        </Token>
    </iOSection>
    <iOSection>
        <Details>           
            <Info>
                <pNumber>FB061690</pNumber>
                <indicator>NO</indicator>
                <Identifier>2</Identifier>
            </Info>
        </Details>
        <Token>
            <Reference>1UUYD05BHM21OJCK3881C7F</Reference>
        </Token>
    </iOSection>
</IResponse>

我在XSL下面尝试过,但没有更近距离

<?xml version="1.0" encoding="UTF-8"?>
<xsl:stylesheet version="1.0" xmlns:xsd="http://www.w3.org/2001/XMLSchema" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:xsl="http://www.w3.org/1999/XSL/Transform"  xmlns:xs="http://www.w3.org/2001/XMLSchema" exclude-result-prefixes="#all" >

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

    <xsl:template match="/*[local-name()='IResponse']/*[local-name()='iOSection']/*[local-name()='Details']/*[local-name()='Info']/*[local-name()='indicator']">

        <xsl:variable name="RSDetails">
            <RS PartID="abcd" SysID="mnc">
                <PDetails>
                    <PN>FB063586</PN>
                    <PStatus>
                        <Code>0</Code>
                        <Description>Cancelled</Description>
                    </PStatus>
                </PDetails>
                <PDetails>
                    <Error>
                        <Code>92</Code>
                        <Message>failed</Message>
                    </Error>
                </PDetails>
            </RS>       
        </xsl:variable> 

        <xsl:element name="indicator">
            <xsl:variable name="PStatus">
                <xsl:value-of select="$RSDetails/RS/PDetails/PStatus" />
            </xsl:variable>
            <xsl:variable name="Message">
                <xsl:value-of select="$RSDetails/RS/PDetails/Message" />
            </xsl:variable>

            <xsl:choose>
                <xsl:when test="$PStatus='0' and $Message='Cancelled'">
                    <xsl:value-of select="'YES'" />
                </xsl:when>
                <xsl:otherwise>
                    <xsl:value-of select="'NO'" />
                </xsl:otherwise>
            </xsl:choose>

        </xsl:element>
        <xsl:copy-of select="@*" />
    </xsl:template>
</xsl:stylesheet>

1 个答案:

答案 0 :(得分:1)

您要在此处执行的操作是查找来自外部文件中匹配节点的值。您没有这么说,但我认为匹配节点是PN值与本地pNumber匹配的节点。

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:strip-space elements="*"/>

<xsl:param name="RSPDetails" select="document('your_other_file.xml')" />
<xsl:key name="rsp" match="PDetails" use="PN" />

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

<xsl:template match="indicator">
    <xsl:copy>
        <xsl:variable name="pnum" select="../pNumber"/>
        <!-- switch context to the other file -->
        <xsl:for-each select="$RSPDetails">
            <xsl:choose>
                <xsl:when test="key('rsp', $pnum)/PStatus/Code=0 and key('rsp', $pnum)/PStatus/Description='Cancelled'">YES</xsl:when>
                <xsl:otherwise>NO</xsl:otherwise>
            </xsl:choose>
        </xsl:for-each>
    </xsl:copy>
</xsl:template>

</xsl:stylesheet>

编辑:

要将查找基于位置,请尝试:

<xsl:template match="indicator">
    <xsl:copy>
        <xsl:variable name="i">
            <xsl:number count="iOSection"/>
        </xsl:variable>
        <xsl:variable name="detail" select="$RSPDetails/RS/PDetails[number($i)]"/>
        <xsl:choose>
            <xsl:when test="$detail/PStatus/Code=0 and $detail/PStatus/Description='Cancelled'">YES</xsl:when>
            <xsl:otherwise>NO</xsl:otherwise>
        </xsl:choose>
    </xsl:copy>
</xsl:template>

在这种情况下,<xsl:key>指令不是必需的。