Десериализовать список <ArrayList>объект - PullRequest
3 голосов
/ 16 февраля 2011

Я пытаюсь десериализовать 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>();
      }
   }

Я могу десериализоваться до одного уровня, то есть я получаю объекты списка элементов, но отдельные объекты элементов не содержат объектов списка блоков. Что я здесь не так делаю?

Ответы [ 2 ]

3 голосов
/ 16 февраля 2011

Измените свой класс 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
0 голосов
/ 16 февраля 2011

Я не знаю точно, что вы здесь делаете, но есть недочёт с десериализацией (по крайней мере, бинарная десериализация. Я не знаю, но я подозреваю, что это то же самое для сериализации XML). При десериализации List<T> или Dictionary<S,T> список заполняется значениями null (или значением по умолчанию, если это тип значения) до выхода из конструктора десериализации. Только после выхода из конструктора список заполняется фактическими десериализованными T s.

Это означает, что если вы хотите что-то сделать со списком, это нельзя сделать в конструкторе. Вместо этого вы можете создать метод, который содержит любую работу, которую необходимо выполнить со списком, и если он аннотирован атрибутом [OnDeserializedAttribute], он будет называться после , список заполнен, но до десериализация возвращается. Метод может иметь любое имя.

Подробнее см. MSDN .

...