重新排序xml,首先是文本节点,然后是带有子节点的节点

时间:2012-06-20 16:23:31

标签: xml xslt xsd

  

可能重复:
  Rebuild magento XML with nodes related to each other, closer together with a transform.

这是一个双重帖子但是使用当前标签我没有得到回复。我会更新其他帖子或删除它。

我从Magento那里得到了回报,开发人员无法轻易改变“后”的例子。我的解析器解析这个有一些问题所以我的问题是。我可以使用xsl样式表将其转换为“after”示例,其中nodea到nodeh更接近彼此,因此更具可读性。

这将为我节省大量时间来调查解析器。

在:

    <Envelope encodingStyle="http://schemas.xmlsoap.org/soap/encoding/">
<Body>
    <salesOrderInfoResponse>
        <result type="ns1:salesOrderEntity">
            <nodec>value</nodec>
            <noded>value</noded>
            <shipping_address type="ns1:salesOrderAddressEntity">
                <parent_id type="xsd:string">762</parent_id>
                <address_type type="xsd:string">shipping</address_type>
                <firstname type="xsd:string">K</firstname>
                <lastname type="xsd:string">Jansen</lastname>
            </shipping_address>
            <billing_address type="ns1:salesOrderAddressEntity">
                <parent_id type="xsd:string">762</parent_id>
                <address_type type="xsd:string">billing</address_type>
                <firstname type="xsd:string">K</firstname>
                <lastname type="xsd:string">Jansen</lastname>
            </billing_address>
            <items arrayType="ns1:salesOrderItemEntity[4]" type="ns1:salesOrderItemEntityArray">
                <item type="ns1:salesOrderItemEntity">
                    <item_id type="xsd:string">3105</item_id>
                </item>
                <item type="ns1:salesOrderItemEntity">
                    <item_id type="xsd:string">3106</item_id>
                </item>
                <item type="ns1:salesOrderItemEntity">
                    <item_id type="xsd:string">3107</item_id>
                </item>
                <item type="ns1:salesOrderItemEntity">
                    <item_id type="xsd:string">3108</item_id>
                </item>
            </items>
            <payment type="ns1:salesOrderPaymentEntity">
                <parent_id type="xsd:string">762</parent_id>
                <cc_last4 type="xsd:string"></cc_last4>
            </payment>
            <nodea>value</nodea>
            <nodeb>value</nodeb>
            <nodee>value</nodee>
            <nodef>value</nodef>
            <nodeg>value</nodeg>
            <nodeh>value</nodeh>
        </result>
    </salesOrderInfoResponse>
</Body>

后:

<Envelope encodingStyle="http://schemas.xmlsoap.org/soap/encoding/">
<Body>
    <salesOrderInfoResponse>
        <result type="ns1:salesOrderEntity">
            <nodea>value</nodea>
            <nodeb>value</nodeb>
            <nodec>value</nodec>
            <noded>value</noded>
            <nodee>value</nodee>
            <nodef>value</nodef>
            <nodeg>value</nodeg>
            <nodeh>value</nodeh>
            <shipping_address type="ns1:salesOrderAddressEntity">
                <parent_id type="xsd:string">762</parent_id>
                <address_type type="xsd:string">shipping</address_type>
                <firstname type="xsd:string">K</firstname>
                <lastname type="xsd:string">Jansen</lastname>
            </shipping_address>
            <billing_address type="ns1:salesOrderAddressEntity">
                <parent_id type="xsd:string">762</parent_id>
                <address_type type="xsd:string">billing</address_type>
                <firstname type="xsd:string">K</firstname>
                <lastname type="xsd:string">Jansen</lastname>
            </billing_address>
            <items arrayType="ns1:salesOrderItemEntity[4]" type="ns1:salesOrderItemEntityArray">
                <item type="ns1:salesOrderItemEntity">
                    <item_id type="xsd:string">3105</item_id>
                </item>
                <item type="ns1:salesOrderItemEntity">
                    <item_id type="xsd:string">3106</item_id>
                </item>
                <item type="ns1:salesOrderItemEntity">
                    <item_id type="xsd:string">3107</item_id>
                </item>
                <item type="ns1:salesOrderItemEntity">
                    <item_id type="xsd:string">3108</item_id>
                </item>
            </items>
            <payment type="ns1:salesOrderPaymentEntity">
                <parent_id type="xsd:string">762</parent_id>
                <cc_last4 type="xsd:string"></cc_last4>
            </payment>
        </result>
    </salesOrderInfoResponse>
</Body>

我添加了一个xslt建议并对其进行了一些修改,它的接缝工作正常:              

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


<xsl:template match="result"> 
    <xsl:copy> 
        <xsl:apply-templates select="@*" /> 
        <xsl:apply-templates select="*[    starts-with(local-name(),'node') ]" /> 
        <xsl:apply-templates select="*[not(starts-with(local-name(),'node'))]|processing-instruction()|comment()" /> 
    </xsl:copy> 
</xsl:template> 

我知道:

  <salesOrderInfoResponse> 

     <result type="ns1:salesOrderEntity">
        <nodec>value</nodec>
        <noded>value</noded>
        <nodea>value</nodea>
        <nodeb>value</nodeb>
        <nodee>value</nodee>
        <nodef>value</nodef>
        <nodeg>value</nodeg>
        <nodeh>value</nodeh>
        <shipping_address type="ns1:salesOrderAddressEntity"> 

           <parent_id type="xsd:string">762</parent_id> 

           <address_type type="xsd:string">shipping</address_type> 

           <firstname type="xsd:string">K</firstname> 

           <lastname type="xsd:string">Jansen</lastname> 

        </shipping_address>
        <billing_address type="ns1:salesOrderAddressEntity"> 

           <parent_id type="xsd:string">762</parent_id> 

           <address_type type="xsd:string">billing</address_type> 

           <firstname type="xsd:string">K</firstname> 

           <lastname type="xsd:string">Jansen</lastname> 

        </billing_address>
        <items arrayType="ns1:salesOrderItemEntity[4]" type="ns1:salesOrderItemEntityArray"> 

           <item type="ns1:salesOrderItemEntity"> 

              <item_id type="xsd:string">3105</item_id> 

           </item> 

           <item type="ns1:salesOrderItemEntity"> 

              <item_id type="xsd:string">3106</item_id> 

           </item> 

           <item type="ns1:salesOrderItemEntity"> 

              <item_id type="xsd:string">3107</item_id> 

           </item> 

           <item type="ns1:salesOrderItemEntity"> 

              <item_id type="xsd:string">3108</item_id> 

           </item> 

        </items>
        <payment type="ns1:salesOrderPaymentEntity"> 

           <parent_id type="xsd:string">762</parent_id> 

           <cc_last4 type="xsd:string"/> 

        </payment>
     </result> 

  </salesOrderInfoResponse> 

这几乎是正确的!我不明白额外的间距来自哪里,但我在正确的轨道上。

使用真实数据进一步测试我没有看到顶部的文本节点。在上面的例子中它是,我的猜测是因为'node'命令。我创建了这个例子并用作文本节点,nodea,nodeb等,但在实际中,它们被称为orderid,name,quantity。

查看带有子节点的xml节点我可以看到它们都有一个属性类型,其值以“ns1:...”开头

1 个答案:

答案 0 :(得分:1)

试试这个XSLT 1.0样式表...

<?xml version="1.0" encoding="utf-8"?>
<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="@*|node()" />
 </xsl:copy>
</xsl:template>

<xsl:template match="result">
 <xsl:copy>
   <xsl:apply-templates select="@*" />
   <xsl:apply-templates select="*[not(starts-with(local-name(),'node'))]|processing-instruction()|comment()" />
   <xsl:apply-templates select="*[    starts-with(local-name(),'node') ]" />
 </xsl:copy>
</xsl:template>

</xsl:stylesheet>

或者,您可以更改...

的模板
<xsl:template match="result">
 <xsl:copy>
   <xsl:apply-templates select="@*" />
   <xsl:apply-templates select="node()">
     <xsl:sort select="starts-with(local-name(),'node') and self::*" data-type="number" />
  </xsl:apply-templates>
 </xsl:copy>
</xsl:template>

如果您需要XSLT 2.0,请告诉我们,因为XSLT 2.0将有更简洁的解决方案。