使用DTO对象中的不同子节点序列化节点

本文关键字:子节点 序列化 节点 DTO 对象 使用 | 更新日期: 2023-09-27 18:27:59

我使用c#和XmlSerializer类从DTO对象创建xml。

现在我必须生成这样的xml文本:

<Order>
    <OrderNo>123456</OrderNo>
    <Positions>
        <TextPosition>
            <Text>This is Order No 123456</Text>
        </TextPosition>
        <ItemPosition>
            <ItemId>14789</ItemId>
            <ItemName>Product 1</ItemName>
        </ItemPosition>
        </TextPosition>
        <ItemPosition>
            <ItemId>456</ItemId>
            <ItemName>Product 2</ItemName>
        </ItemPosition>
        <TextPosition>
            <Text>Good bye</Text>
        </TextPosition>
        <SumPosition>
            <Value>123.45 USD</Value>
        </SumPosition>
    </Positions>
</Order>

我使用属性来装饰我的类,一切都很好。有一件事我还不能解决。我需要生成一个标签Positions,里面有不同的标签TextPosition, ItemPosition, ValuePosition, ...

我如何在c#中实现这一点?

目前我的班级订单包含

[XmlElement("Positions")]
public PositionList Positions { get; set; }

PositionList是一个具有的类

public class PositionList
{
    [XmlElement("Positions")]
    public List<Object> Positions { get; set; }
}

为了避免InvalidOperationException,我添加了

[XmlInclude(typeof(Textposition))]
[XmlInclude(typeof(ItemPosition))]
[XmlInclude(typeof(SumPosition))]

我的订单课。

然而,不是生成

<Positions>
    <TextPosition>...</TextPosition>
</Positions>

串行器生成:

<Position d4p1:type="TextPosition" xmlns:d4p1="http://www.w3.org/2001/XMLSchema-instance">...</Position>

问题是,In不需要自己反序列化它,而是为一个客户提供一个非常严格格式的XML文件。有办法做到这一点吗?

使用DTO对象中的不同子节点序列化节点

有一种方法:

public class Order
{
   public List<BasePosition> Positions { get; set; }
   public Order() { Positions = new List<BasePosition>(); }
}
public class BasePosition
{
}
public class TextPosition : BasePosition
{
   public string Text { get; set; }
}
public class ItemPosition : BasePosition
{
   public int ItemId { get; set; }
   public string ItemName { get; set; }
}
public class SumPosition : BasePosition
{
   public string Value { get; set; }
}

将其序列化为XML文件的示例:

Order o = new Order();
o.Positions.Add(new TextPosition() { Text = "This is Order No 123456" });
o.Positions.Add(new ItemPosition() { ItemId = 14789, ItemName = "Product 1" });
o.Positions.Add(new TextPosition());
o.Positions.Add(new ItemPosition() { ItemId = 456, ItemName = "Product 2" });
o.Positions.Add(new SumPosition() { Value = "123.45 USB" });
XmlAttributeOverrides specific_attributes = new XmlAttributeOverrides();
XmlAttributes attrs = new XmlAttributes();
attrs.XmlElements.Add(new XmlElementAttribute(typeof(TextPosition)));
attrs.XmlElements.Add(new XmlElementAttribute(typeof(ItemPosition)));
attrs.XmlElements.Add(new XmlElementAttribute(typeof(SumPosition)));
specific_attributes.Add(typeof(Order), "Positions", attrs);
XmlSerializer ser = new XmlSerializer(typeof(Order), specific_attributes);
using(MemoryStream mem_stream = new MemoryStream())
{
   ser.Serialize(mem_stream, o);
   using (BinaryWriter bw = new BinaryWriter(new FileStream("orders.xml", FileMode.Create)))
   {
      bw.Write(mem_stream.ToArray());
   }
}

结果:

<?xml version="1.0"?>
<Order xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:xsd="http://www.w3.org/2001/XMLSchema">
  <TextPosition>
    <Text>This is Order No 123456</Text>
  </TextPosition>
  <ItemPosition>
    <ItemId>14789</ItemId>
    <ItemName>Product 1</ItemName>
  </ItemPosition>
  <TextPosition />
  <ItemPosition>
    <ItemId>456</ItemId>
    <ItemName>Product 2</ItemName>
  </ItemPosition>
  <SumPosition>
    <Value>123.45 USB</Value>
  </SumPosition>
</Order>

当您需要非常严格格式的XML时,使用序列化程序可能不是最好的选择。有了序列化程序,为了方便而放弃了控制权,而现在您正在努力夺回控制权。

考虑使用Linq-to-XML,即XDocument类。

var doc = 
    new XElement("Order", 
      new XElement ("OrderNo", 123456),
      new XElement ("Positions", 
         myPositions.Select(p => new XElement("Position", .... ) ) );