DataContractSerializer和泛型列表

本文关键字:列表 泛型 DataContractSerializer | 更新日期: 2023-09-27 18:12:03

我有问题从我的xml文件反序列化列表。我的属性是内部的,所以我使用的是datacontractserializer,而不是xmlserializer。

我的xml文件如下

<?xml version="1.0" encoding="utf-8" ?>
<Root xmlns:i="http://www.w3.org/2001/XMLSchema-instance" xmlns="http://schemas.datacontract.org/2004/07/Root">
    <BackgroundColor>
        0,0,255
    </BackgroundColor>
    <RowGap>1</RowGap>
    <Table>
        <TableHeading>"H1"</TableHeading>
        <StartingColumn>A</StartingColumn>
    </Table>
    <Table>
        <TableHeading>"H2"</TableHeading>
        <StartingColumn>B</StartingColumn>
    </Table>    
</Root>

Root有List of Table。ListOfTables没有被反序列化。类是

[DataContract()]
public class Root
{
    [OnDeserialized]
    internal void OnSerializingMethod(StreamingContext context)
    {
        if (Table == null)
        {
            Table = new List<TableStructure>();
        }
     }
     [DataMember(Name = "RowGap")]
     internal int RowGap { get; set; }
     [DataMember(Name = "TableHeaderBackgroundColor")]
     internal string HdrBackColor  { get; set; }
     [DataMember()]
     internal List<TableStructure> Table { get; set; }
}
[DataContract(Name = "Table", Namespace = "")]
public sealed class TableStructure 
{
    [DataMember]
    public string TableHeading { get; set; }
    [DataMember]
    public string StartingColumn { get; set; }
}

描述方法

internal static Root GetSettings()
{
     Root settings;
     using (FileStream fs = new FileStream("Root.xml", FileMode.Open, FileAccess.Read))
     {
          using (XmlDictionaryReader reader = XmlDictionaryReader.CreateTextReader(fs, new XmlDictionaryReaderQuotas()))
          {
              var x = new DataContractSerializer(typeof(Root));
              settings = (Root)x.ReadObject(reader);
              reader.Close();
              fs.Close();
          }
     }
     return settings;
}

执行没有任何错误,但不加载任何表信息。

DataContractSerializer和泛型列表

我修改了您的解决方案以使其工作(添加了XML名称空间):

[DataContract(Namespace = "http://schemas.datacontract.org/2004/07/Root")]
public class Root
{
    [OnDeserialized]
    internal void OnSerializingMethod(StreamingContext context)
    {
        if (Table == null)
        {
            Table = new List<TableStructure>();
        }
    }
    [DataMember(Name = "RowGap")]
    internal int RowGap { get; set; }
    [DataMember(Name = "TableHeaderBackgroundColor")]
    internal string HdrBackColor { get; set; }
    [DataMember(Name = "Tables")]
    internal List<TableStructure> Table { get; set; }
}
[DataContract(Name = "Table", Namespace = "http://schemas.datacontract.org/2004/07/Root")]
public sealed class TableStructure
{
    [DataMember(Name = "StartingColumn")]
    public string TableHeading { get; set; }
    [DataMember(Name = "TableHeading")]
    public string StartingColumn { get; set; }
}
XML文件:

<?xml version="1.0" encoding="utf-8" ?>
<Root xmlns:i="http://www.w3.org/2001/XMLSchema-instance" xmlns="http://schemas.datacontract.org/2004/07/Root">
  <BackgroundColor>
    0,0,255
  </BackgroundColor>
  <RowGap>1</RowGap>
  <!-- Used as container element for list -->
  <Tables>
    <Table>
      <!-- Sorted elements alphabetically -->
      <StartingColumn>A</StartingColumn>
      <TableHeading>"H1"</TableHeading>
    </Table>
    <Table>
      <!-- Sorted elements alphabetically -->
      <StartingColumn>B</StartingColumn>
      <TableHeading>"H2"</TableHeading>
    </Table>
  </Tables>
</Root>

注意: XML元素的顺序由DataContractSerializer考虑,这里提到:http://social.msdn.microsoft.com/Forums/vstudio/en-US/a891928b-d27a-4ef2-83b3-ee407c6b9187/order-of-data-members-in-the-xml-string-influences-deserialization-datacontractserializer?forum=wcf。由于没有按字母顺序对元素进行排序,所以我修改了XML文件以满足DataContractSerializer的行为。

可以使用DataMember属性及其DataMember来指定XML元素的顺序。顺序属性。

您是如何创建xml的?手工还是通过序列化器?在现有的<Table>元素(代表每个TableStructure实例)周围应该有一个额外的<Table>元素(代表Root.Table属性)。

这个xml(由DataContractSerializer创建)在我的机器上按预期反序列化:

<Root xmlns="http://schemas.datacontract.org/2004/07/TestDataGrid" xmlns:i="http://www.w3.org/2001/XMLSchema-instance">
    <RowGap>101</RowGap>
    <Table>
        <Table xmlns="">
            <StartingColumn>1</StartingColumn>
            <TableHeading>First</TableHeading>
        </Table>
        <Table xmlns="">
            <StartingColumn>2</StartingColumn>
            <TableHeading>Second</TableHeading>
        </Table>
    </Table>
    <TableHeaderBackgroundColor>c0ffee</TableHeaderBackgroundColor>
</Root>