子模板中的XSLT复制父元素树

时间:2012-10-08 11:02:41

标签: xml xslt xslt-2.0

嗯,例如我有一个XML:

<ProcessPurchaseOrder >
    <PurchaseOrderLine>
        <DocumentReference type="sendersReference2">
            <DocumentID>
                <ID>100</ID>
            </DocumentID>
        </DocumentReference>
        <DocumentReference type="sendersReference3">
            <DocumentID>
                <ID>ru</ID>
            </DocumentID>
        </DocumentReference>
        <Item>
            <CustomerItemID>
                <ID>00126</ID>
            </CustomerItemID>
        </Item>
    </PurchaseOrderLine>
    <PurchaseOrderLine>
        <DocumentReference type="sendersReference2">
            <DocumentID>
                <ID>200</ID>
            </DocumentID>
        </DocumentReference>
        <DocumentReference type="sendersReference3">
            <DocumentID>
                <ID>ru</ID>
            </DocumentID>
        </DocumentReference>
        <Item>
            <CustomerItemID>
                <ID>123122</ID>
            </CustomerItemID>
        </Item>
    </PurchaseOrderLine>
</ProcessPurchaseOrder>

和XSLT的一部分:

 <xsl:for-each select="*:PurchaseOrderLine">
    <xsl:variable name="ArtNr" select="*:Item/*:CustomerItemID/*:ID"/>
    <xsl:variable name="WepNr" select="/*/DbResponse/ResultSet/Row[Cell[@name='ARTNR']=$ArtNr][Cell[@name='WEANR']=$WeaNr]/Cell[@name='WEPNR']"/>
    <xsl:copy>
        <xsl:if test="$WepNr!=''">
            <xsl:for-each select="$WepNr">
                <LineNumber><xsl:value-of select="$WepNr/current()"/></LineNumber>
            </xsl:for-each>
        </xsl:if>
        <xsl:apply-templates select="@*|node()"/>
    </xsl:copy>
</xsl:for-each>

对于每个$WepNr值,我要复制整个<PurchaseOrderLine>并插入<LineNumber>值为当前WepNr

因此,例如:如果第一个PurchaseOrderLine的$WepNr返回:16;结果将是:

<ProcessPurchaseOrder >
    <PurchaseOrderLine>
    <!-- wepnr[1] = 16 -->
    <LineNumber>16</LineNumber>
        <DocumentReference type="sendersReference2">
            <DocumentID>
                <ID>100</ID>
            </DocumentID>
        </DocumentReference>
        <DocumentReference type="sendersReference3">
            <DocumentID>
                <ID>ru</ID>
            </DocumentID>
        </DocumentReference>
        <Item>
            <CustomerItemID>
                <ID>00126</ID>
            </CustomerItemID>
        </Item>
    </PurchaseOrderLine>
    <PurchaseOrderLine>
    <!-- copied PurchaseOrderLine with wepnr[2]=26 -->
    <LineNumber>26</LineNumber>
        <DocumentReference type="sendersReference2">
            <DocumentID>
                <ID>100</ID>
            </DocumentID>
        </DocumentReference>
        <DocumentReference type="sendersReference3">
            <DocumentID>
                <ID>ru</ID>
            </DocumentID>
        </DocumentReference>
        <Item>
            <CustomerItemID>
                <ID>00126</ID>
            </CustomerItemID>
        </Item>
    </PurchaseOrderLine>

    <!-- here is 2nd PurchaseOrderLine>
    <!-- ... -->

</ProcessPurchaseOrder>

有可能吗?

UPD : DBresponse XML部分

<DbResponse>
   <ResultSet>
      <Row>
         <Cell name="WEANR" type="VARCHAR2">1909123</Cell>
         <Cell name="ARTNR" type="VARCHAR2">00126</Cell>
         <Cell name="WEPNR" type="VARCHAR2">1</Cell>
      </Row>
      <Row>
         <Cell name="WEANR" type="VARCHAR2">1909123</Cell>
         <Cell name="ARTNR" type="VARCHAR2">00126</Cell>
         <Cell name="WEPNR" type="VARCHAR2">16</Cell>
      </Row>
   </ResultSet>
</DbResponse>

这只是意味着WepNr可以返回多个值:例如“1 16”在这种情况下

1 个答案:

答案 0 :(得分:1)

我认为不是

 <xsl:for-each select="*:PurchaseOrderLine">
    <xsl:variable name="ArtNr" select="*:Item/*:CustomerItemID/*:ID"/>
    <xsl:variable name="WepNr" select="/*/DbResponse/ResultSet/Row[Cell[@name='ARTNR']=$ArtNr][Cell[@name='WEANR']=$WeaNr]/Cell[@name='WEPNR']"/>
    <xsl:copy>
        <xsl:if test="$WepNr!=''">
            <xsl:for-each select="$WepNr">
                <LineNumber><xsl:value-of select="$WepNr/current()"/></LineNumber>
            </xsl:for-each>
        </xsl:if>
        <xsl:apply-templates select="@*|node()"/>
    </xsl:copy>
</xsl:for-each>

你更想要:

 <xsl:apply-templates select="*:PurchaseOrderLine"/>

然后在顶层

 <xsl:key name="row-by-wepnr" 
   match="DbResponse/ResultSet/Row"
   use="Cell[@name='ARTNR']"/>


 <xsl:template match="*:PurchaseOrderLine">
   <xsl:variable name="this" select="."/>
   <xsl:variable name="wepNrs" select="key('row-by-wepnr', *:Item/*:CustomerItemID/*:ID)[Cell[@name='WEANR']=$WeaNr]/Cell[@name='WEPNR']"/>
   <xsl:for-each select="$wepNrs">
     <xsl:apply-templates select="$this" mode="add-wep">
       <xsl:with-param name="wep" select="current()"/>
     </xsl:apply-templates>
   </xsl:for-each>
 </xsl:template>

 <xsl:template match="*:PurchaseOrderLine" mode="add-wep">
    <xsl:param name="wep"/>
    <xsl:copy>
      <LineNumber><xsl:value-of select="$wep"/></LineNumber>
      <xsl:apply-templates select="@*|node()"/>
    </xsl:copy>
</xsl:template>

假设$WearNr是一个全局参数或变量,否则你需要在例如传递该值。

 <xsl:apply-templates select="*:PurchaseOrderLine">
   <xsl:with-param name"WearNr" select="$WearNr"/>
  </xsl:apply-templates>

 <xsl:template match="*:PurchaseOrderLine">
   <xsl:param name="$WearNr"/>
   <xsl:variable name="this" select="."/>
   <xsl:variable name="wepNrs" select="key('row-by-wepnr', *:Item/*:CustomerItemID/*:ID)[Cell[@name='WEANR']=$WeaNr]/Cell[@name='WEPNR']"/>
   <xsl:for-each select="$wepNrs">
     <xsl:apply-templates select="$this" mode="add-wep">
       <xsl:with-param name="wep" select="current()"/>
     </xsl:apply-templates>
   </xsl:for-each>
 </xsl:template>

 <xsl:template match="*:PurchaseOrderLine" mode="add-wep">
    <xsl:param name="wep"/>
    <xsl:copy>
      <LineNumber><xsl:value-of select="$wep"/></LineNumber>
      <xsl:apply-templates select="@*|node()"/>
    </xsl:copy>
</xsl:template>

当然,所有未经测试的,您最好提供最小但完整的代码示例,以便我们编写可测试的代码。