2010-01-13 129 views
2

我只需要以下文件的<Transaction-Detail>节点(由<tran-id>子节点)进行排序:排序只有特定的XML节点

<TransActDO clear="true" removed="false"> 
    <stmt-reason-code>1001</stmt-reason-code> 
    <Transaction-Detail clear="true" removed="false"> 
    <txn-amt>788.20</txn-amt> 
    <txn-description>New Purchase</txn-description> 
    <tran-id>3271</tran-id> 
    </Transaction-Detail> 
    <Transaction-Detail clear="true" removed="false"> 
    <txn-amt>-68.20</txn-amt> 
    <txn-description>Return</txn-description> 
    <tran-id>27795</tran-id> 
    </Transaction-Detail> 
    <Transaction-Detail clear="true" removed="false"> 
    <txn-amt>0.00</txn-amt> 
    <txn-description>Comment</txn-description> 
    <transaction-reason-desc>No Reason</transaction-reason-desc> 
    <tran-id>13365</tran-id> 
    <transaction-reason-code>0</transaction-reason-code> 
    </Transaction-Detail> 
    <Transaction-Detail clear="true" removed="false"> 
    <txn-amt>343.45</txn-amt> 
    <txn-description>New Purchase</txn-description> 
    <tran-id>7558</tran-id> 
    </Transaction-Detail> 
    <Transaction-Detail clear="true" removed="false"> 
    <txn-amt>0.00</txn-amt> 
    <txn-description>Comment</txn-description> 
    <transaction-reason-desc>No Reason</transaction-reason-desc> 
    <tran-id>6512</tran-id> 
    <transaction-reason-code>0</transaction-reason-code> 
    </Transaction-Detail> 
    <account-no>123456789</account-no> 
    <payer-name>JOHN DOE</payer-name> 
    <Product-Detail clear="true" removed="false"> 
    <Name>WIDGET</Name> 
    <Amount>89.00</Amount> 
    </Product-Detail> 
    <Product-Detail clear="true" removed="false"> 
    <Name>NEWER WIDGET</Name> 
    <Amount>99.99</Amount> 
    </Product-Detail> 
    <stmt-reason-desc>Web Statement</stmt-reason-desc> 
    <type>Original</type> 
</TransActDO> 

输出是XML,需要同时复制所有其他节点和属性原始文件。实质上,复制所有内容,只需对Transaction-Detail节点进行排序即可。

我走了这么远:

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

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

    <xsl:template match="TransActDO"> 
    <xsl:copy> 
     <xsl:apply-templates select="@*"/> 
     <xsl:apply-templates select="Transaction-Detail"> 
     <xsl:sort select="tran-id" data-type="number" order="ascending"/> 
     </xsl:apply-templates> 
    </xsl:copy> 
    </xsl:template> 

</xsl:stylesheet> 

这将导致一个适当的排序只包含事务的详细信息节点及其子节点的XML文件。当我试图添加额外的逻辑来复制其余的节点时,排序就会中断。

我想我很难围绕XSLT执行理论和语法包装我的大脑。

任何帮助非常感谢!
-nth-

+0

您需要一个preceed设定事务的详细信息节点的节点保持在顶部和那些遵循保持在底部或付出更多的交易,详细的节点移动到XMl的底部? (顺便说一句,我假设由于遗留的原因,你不能只重新设计XML,使产品详细信息和交易详情每个都有一个包含父节点,所以他们没有不同的标记名的兄弟姐妹) – AnthonyWJones 2010-01-13 22:10:37

回答

4

这将做到这一点: -

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

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

    <xsl:template match="TransActDO"> 
     <xsl:copy> 
      <xsl:apply-templates select="@*|node()[not(preceding-sibling::Transaction-Detail) and not(self::Transaction-Detail)]"/> 
      <xsl:apply-templates select="Transaction-Detail"> 
       <xsl:sort select="tran-id" data-type="number" order="ascending"/> 
      </xsl:apply-templates> 
      <xsl:apply-templates select="@*|node()[not(following-sibling::Transaction-Detail) and not(self::Transaction-Detail)]"/> 
    </xsl:copy> 
    </xsl:template> 


</xsl:stylesheet> 

如果你不介意的移动事务的细节元素的TransActDO元素的顶部或者底部,你可以简化内部设置的应用模板: -

  <xsl:apply-templates select="@*|node()[not(self::Transaction-Detail)]"/> 
      <xsl:apply-templates select="Transaction-Detail"> 
       <xsl:sort select="tran-id" data-type="number" order="ascending"/> 
      </xsl:apply-templates> 
+0

+1 。我刚刚注意到,你已经发布了我刚刚发布的*完全相同的*代码(猜我应该读你的第二个代码示例...)。删除我的答案,以避免冗余。 – Tomalak 2010-01-14 12:55:23

+0

谢谢安东尼。它确实有效!不幸的是,我不能控制xml结构的创建,但至少可以解决排序问题。 – nth 2010-01-14 15:46:10