4

我正在尝试将 XML 反序列化为对象,但我陷入了一种情况。任何人都可以在这里帮助我。

XML:

<?xml version="1.0" ?>
<Level>
  <Warp_Blocks>
        <Warp_Block>
            <Block row="7" col="7" />
            <Block row="2" col="7" />
        </Warp_Block>
        <Warp_Block>
            <Block row="4" col="4" />
            <Block row="3" col="7" />
        </Warp_Block>
  </Warp_Blocks>
</Level>

代码:

  [XmlRoot("Level")]
   public class LData
    {
        [XmlArray("Warp_Blocks")]
        [XmlArrayItem("Warp_Block",typeof(WarpBlock),IsNullable = false)]
        public List<WarpBlock> WarpBlocks;
    }
   public class LBlock
   {
      [XmlAttribute("row")]
      public int row;
      [XmlAttribute("col")]
      public int col;
   }
   public class WarpBlock
   {
      [XmlArray("Warp_Block")]
      [XmlArrayItem("Block",typeof(LBlock),IsNullable= false)]
      public List<LBlock> WarpBlocks;

      public WarpBlock()
      {
            WarpBlocks = new List<LBlock>();
      }
   }

我能够反序列化到一个级别,即我得到一个项目对象列表,但单个项目对象不包含块对象列表。我在这里做错了什么?

4

2 回答 2

4

将您的 LData 类更改为:

[XmlRoot("Level")]
public class LData
{
    [XmlElement("Warp_Blocks")]
    public List<WarpBlock> WarpBlocks;
}

编辑:
我不知道为什么它没有读取您的第二个 Warp_Block。我认为唯一可能的原因可能是您正在做的事情与您在问题中发布的内容不同。这是完整的示例:

[XmlRoot("Level")]
public class LData
{
    [XmlElement("Warp_Blocks")]
    public List<WarpBlock> WarpBlocks;
}
public class LBlock
{
    [XmlAttribute("row")]
    public int row;
    [XmlAttribute("col")]
    public int col;
}
public class WarpBlock
{
    [XmlArray("Warp_Block")]
    [XmlArrayItem("Block", typeof(LBlock), IsNullable = false)]
    public List<LBlock> WarpBlocks;

    public WarpBlock()
    {
        WarpBlocks = new List<LBlock>();
    }
}
public class Program
{
    public static void Main()
    {
        string test =
            "<?xml version=\"1.0\" ?>" +
            "<Level>" +
            "  <Warp_Blocks>" +
            "        <Warp_Block>" +
            "            <Block row=\"7\" col=\"7\" />" +
            "            <Block row=\"2\" col=\"7\" />" +
            "        </Warp_Block>" +
            "        <Warp_Block>" +
            "            <Block row=\"4\" col=\"4\" />" +
            "            <Block row=\"3\" col=\"7\" />" +
            "        </Warp_Block>" +
            "  </Warp_Blocks>" +
            "</Level>";

        byte[] byteArray = Encoding.ASCII.GetBytes(test);
        MemoryStream stream = new MemoryStream(byteArray);

        XmlSerializer s = new XmlSerializer(typeof (LData));
        LData data = (LData) s.Deserialize(stream);

        foreach (var a in data.WarpBlocks)
            foreach (var b in a.WarpBlocks)
                Console.WriteLine(b.row + ", " + b.col);

        Console.ReadKey();
    }
}

它正确输出:

7, 7
2, 7
4, 4
3, 7
于 2011-02-16T13:46:06.437 回答
0

我不确切知道您在这里做什么,但是反序列化有一个问题(至少是二进制反序列化。我不知道,但我怀疑 XML 序列化也是如此)。在反序列化 aList<T>或 aDictionary<S,T>时,列表将填充null值(如果它是值类型,则为默认值),直到退出反序列化构造函数。只有在退出构造函数后,列表才会填充实际的反序列化Ts。

这意味着如果你想对列表做一些事情,它不能在构造函数中完成。您可以改为创建一个包含需要对列表完成的任何工作的方法,并且只要使用[OnDeserializedAttribute]属性对其进行注释,它将在列表被填充之后但反序列化返回之前被调用。该方法可以有任何名称。

有关详细信息,请参阅MSDN

于 2011-02-16T13:01:52.540 回答