Deserializing List <int> с XmlSerializer, вызывающим лишние элементы

Я замечаю странное поведение в XmlSerializer и общих списках (в частности List<int>). Мне было интересно, видел ли кто-нибудь это раньше или знает, что происходит. Похоже, сериализация работает нормально, но десериализация хочет добавить дополнительные элементы в список. Код ниже демонстрирует проблему.

Сериализуемый класс:

public class ListTest
{
    public int[] Array { get; set; }
    public List<int> List { get; set; }

    public ListTest()
    {
        Array = new[] {1, 2, 3, 4};
        List = new List<int>(Array);
    }
}

Тестовый код:

ListTest listTest = new ListTest();
Debug.WriteLine("Initial Array: {0}", (object)String.Join(", ", listTest.Array));
Debug.WriteLine("Initial List: {0}", (object)String.Join(", ", listTest.List));

XmlSerializer serializer = new XmlSerializer(typeof(ListTest));
StringBuilder xml = new StringBuilder();
using(TextWriter writer = new StringWriter(xml))
{
    serializer.Serialize(writer, listTest);
}

Debug.WriteLine("XML: {0}", (object)xml.ToString());

using(TextReader reader = new StringReader(xml.ToString()))
{
    listTest = (ListTest) serializer.Deserialize(reader);
}

Debug.WriteLine("Deserialized Array: {0}", (object)String.Join(", ", listTest.Array));
Debug.WriteLine("Deserialized List: {0}", (object)String.Join(", ", listTest.List));

Отладочный вывод:

Initial Array: 1, 2, 3, 4
Initial List: 1, 2, 3, 4

XML:

<?xml version="1.0" encoding="utf-16"?>
<ListTest xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:xsd="http://www.w3.org/2001/XMLSchema">
  <Array>
    <int>1</int>
    <int>2</int>
    <int>3</int>
    <int>4</int>
  </Array>
  <List>
    <int>1</int>
    <int>2</int>
    <int>3</int>
    <int>4</int>
  </List>
</ListTest>
Deserialized Array: 1, 2, 3, 4
Deserialized List: 1, 2, 3, 4, 1, 2, 3, 4

Обратите внимание, что как массив, так и список, похоже, правильно сериализованы в XML, но при десериализации массив получается корректно, но список возвращается с дублированным набором элементов. Любые идеи?

Ответ 1

Это происходит потому, что вы инициализируете List в конструкторе. Когда вы переходите к десериализации, создается новый ListTest, а затем он заполняет объект из состояния.

Подумайте о рабочем процессе, подобном этому

  • Создайте новый ListTest
  • Выполнить конструктор (добавить 1,2,3,4)
  • Отключить состояние xml и добавить 1,2,3,4 в список

Простым решением было бы инициировать объект вне рамки конструктора.

public class ListTest
{
    public int[] Array { get; set; }
    public List<int> List { get; set; }

    public ListTest()
    {

    }

    public void Init() 
    {
        Array = new[] { 1, 2, 3, 4 };
        List = new List<int>(Array);
    }
}

ListTest listTest = new ListTest();
listTest.Init(); //manually call this to do the initial seed

Ответ 2

Проблема заключается в том, что вы определяете оригинал 1,2,3,4 в списке в конструкторе по умолчанию. Ваш десериализатор добавляет в список, а не определяет его.