xslt分组 - 从一种xml格式转换为另一种格式

时间:2012-04-27 10:01:52

标签: c# datacontractserializer xml-deserialization

我有一个xslt,用于从一种xml格式转换为另一种格式。

我的源文件有多个“生产订单”和一个“销售订单” 我的目的地只有一个“销售订单”,其中包含所有生产订单。

这是我的源xml格式

 <GenomicOrder>
  <SalesOrder>
    ....
  </SalesOrder>
  <ProductionOrder>
    ....
  </ProductionOrder>
  <ProductionOrder>
    ....
  </ProductionOrder>
  <ProductionOrder>
    ....
  </ProductionOrder>
</GenomicOrder>

我的目的地xml格式如下

<GenomicOrder>
  <SalesOrder>
    ....
  <ProductionOrder>
    ....
  </ProductionOrder>
  <ProductionOrder>
    ....
  </ProductionOrder>
  <ProductionOrder>
    ....
  </ProductionOrder>
  </SalesOrder>
</GenomicOrder>

我的反序列化代码如下

DataContractSerializer ser = new DataContractSerializer(typeof(SalesOrder));
            FileStream fs =
                new FileStream(
                    @"C:\realsample.xml",
                    FileMode.Open);
            XmlDictionaryReader reader1 = XmlDictionaryReader.CreateTextReader(fs, new XmlDictionaryReaderQuotas());

            //Deserialize the data and read it from the instance.
            SalesOrder parent = (SalesOrder)ser.ReadObject(reader1, true);

            reader1.Close();

我的成功销售订单xslt转换如下

<xsl:template match="/">
    <SalesOrder>
      <xsl:apply-templates select="GenomicOrder/SalesOrder"/>
      <ProductionOrders>
        <xsl:apply-templates select="GenomicOrder/ProductionOrder"/>
      </ProductionOrders>
    </SalesOrder>
  </xsl:template>

SalesOrder.OrderId = 1234; SalesOrder.ProductionOrder = null;

我的成功生产订单xslt转换如下

<xsl:template match="/">
        <SalesOrder>
          <ProductionOrders>
            <xsl:apply-templates select="GenomicOrder/ProductionOrder"/>
          </ProductionOrders>
          <xsl:apply-templates select="GenomicOrder/SalesOrder"/>
        </SalesOrder>
</xsl:template>

SalesOrder.OrderId = NULL; SalesOrder.ProductionOrder.OrderRef = 12345;

我是否可以对一次传递中的所有对象进行反序列化。我为“GenomicOrder / ProductionOrder”和“GenomicOrder / SalesOrder”编写了转换。

1 个答案:

答案 0 :(得分:1)

我不确定我是否理解这个问题,假设你想要在SalesOrder元素之后包含所有元素,你可以使用像

这样的方法
<xsl:stylesheet
  xmlns:xsl="http://www.w3.org/1999/XSL/Transform"
  version="2.0">

  <xsl:strip-space elements="*"/>
  <xsl:output indent="yes"/>

  <xsl:template match="GenomicOrder">
    <xsl:copy>
      <xsl:for-each-group select="*" group-starting-with="SalesOrder">
        <SalesOrder>
          <xsl:copy-of select="node(), current-group() except ."/>
        </SalesOrder>
      </xsl:for-each-group>
    </xsl:copy>
  </xsl:template>

</xsl:stylesheet>

XSLT 2.0样式表,在输入上使用Saxon 9.4运行时

 <GenomicOrder>
  <SalesOrder>
    ....
  </SalesOrder>
  <ProductionOrder>
    ....
  </ProductionOrder>
  <ProductionOrder>
    ....
  </ProductionOrder>
  <ProductionOrder>
    ....
  </ProductionOrder>
</GenomicOrder>

输出

<GenomicOrder>
   <SalesOrder>
    ....
  <ProductionOrder>
    ....
  </ProductionOrder>
      <ProductionOrder>
    ....
  </ProductionOrder>
      <ProductionOrder>
    ....
  </ProductionOrder>
   </SalesOrder>
</GenomicOrder>

如果我的建议无效,您可能需要在XML输入结构上发布更多详细信息。