XmlSerializer и List <T>со значениями по умолчанию - PullRequest
12 голосов
/ 18 ноября 2010

Я наблюдал странное поведение при сериализации и десериализации класса, который имеет член типа List<T>, который был заполнен значениями по умолчанию во время создания. В отличие от свойства на основе массива, свойство типа List<T> не будет очищено при десериализации XmlSerializer.

Вот мой код:

public class Program
{
    public class Config
    {
        public Config()
        {
            Test1 = new List<string>()  {"A", "B"};
            Test2 = new String[] {"A", "B"};
        }
        public List<string> Test1 {get;set;}
        public string[] Test2 {get;set;}
    }

    public static void Main()
    {
        XmlSerializer xmlSerializer =
            new XmlSerializer(typeof(Config));
        using(Stream s = new MemoryStream())
        {
            xmlSerializer.Serialize(s, new Config());
            s.Position = 0;
            xmlSerializer.Serialize(Console.Out,
                xmlSerializer.Deserialize(s));
        }
    }
}

И это вывод:

<?xml version="1.0" encoding="ibm850"?>
<Config xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    xmlns:xsd="http://www.w3.org/2001/XMLSchema">
  <Test1>
    <string>A</string>
    <string>B</string>
    <string>A</string>
    <string>B</string>
  </Test1>
  <Test2>
    <string>A</string>
    <string>B</string>
  </Test2>
</Config>

Почему List<T> обрабатывается XmlSerializer иначе, чем массив, и что я могу сделать, чтобы изменить это поведение?

Ответы [ 2 ]

5 голосов
/ 18 ноября 2010

Интересный;Я никогда не замечал этого в прошлом, но это определенно воспроизводимо.Поскольку XmlSerializer не поддерживает обратные вызовы сериализации (чтобы вы знали, что он работает для сериализации), на это трудно повлиять;возможно, самый простой ответ - «не помещайте данные по умолчанию в объекты в конструкторе» (хотя, возможно, предложите фабричный метод, который делает это).

Вы можете попытаться реализовать IXmlSerializable, но это слишком сложно сделать правильно, даже для простого примера.

Я проверил, однако, и DataContractSerializer не ведет себя таким образом - так что вы могли бы, возможно, переключиться наDataContractSerializer;вот мой тестовый код с DCS:

DataContractSerializer ser =
    new DataContractSerializer(typeof(Config));
using (Stream s = new MemoryStream())
{
    ser.WriteObject(s, new Config());
    s.Position = 0;
    using(var writer = XmlWriter.Create(Console.Out)) {
        ser.WriteObject(writer, ser.ReadObject(s));
    }
}

и вот что я имею в виду под заводским методом:

public class Config
{
    public Config()
    {
        Test1 = new List<string>();
        Test2 = nix;
    }
    public List<string> Test1 { get; set; }
    public string[] Test2 { get; set; }

    private static readonly string[] nix = new string[0];
    public static Config CreateDefault()
    {
        Config config = new Config();
        config.Test1.Add("A");
        config.Test1.Add("B");
        config.Test2 = new string[2] { "A", "B" };
        return config;
    }
}
3 голосов
/ 15 августа 2011

Это действительно расстраивает поведение десериализации XML, когда списки содержат набор записей по умолчанию, созданных в конструкторе.

Мой обходной путь состоял в том, чтобы установить XMLIgnoreAttribute в Списке и включить открытый член массива типа объекта, когда set / get обрабатывает заполнение списка из массива.

Что-то вроде следующего позволяет создавать значения по умолчанию в конструкторе, но не позволяет сериализатору XML добавлять записи в список по умолчанию. (ошибки / нулевые проверки в стороне).

public class Config
{
    public Config()
    {
        Test1 = new List<string>() { "A", "B" }; 
        Test2 = new String[] { "A", "B" };

    }

    [XmlIgnore]
    public List<string> Test1 { get; set; }
    public string[] Test2 { get; set; }

    // This member is only to be used during XML serialization
    public string[] Test1_Array 
    { 
        get
        {
            return Test1.ToArray();
        }
        set 
        {
            Test1 = value.ToList();
        }
    }        

}
...