XML-сериализация списка‹T› — корень XML

Первый вопрос о Stackoverflow (.Net 2.0):

Итак, я пытаюсь вернуть XML списка со следующим:

public XmlDocument GetEntityXml()
    {        
        StringWriter stringWriter = new StringWriter();
        XmlDocument xmlDoc = new XmlDocument();            

        XmlTextWriter xmlWriter = new XmlTextWriter(stringWriter);

        XmlSerializer serializer = new XmlSerializer(typeof(List<T>));

        List<T> parameters = GetAll();

        serializer.Serialize(xmlWriter, parameters);

        string xmlResult = stringWriter.ToString();

        xmlDoc.LoadXml(xmlResult);

        return xmlDoc;
    }

Теперь это будет использоваться для нескольких объектов, которые я уже определил.

Скажем, я хотел бы получить XML List<Cat>

XML будет примерно таким:

<ArrayOfCat>
  <Cat>
    <Name>Tom</Name>
    <Age>2</Age>
  </Cat>
  <Cat>
    <Name>Bob</Name>
    <Age>3</Age>
  </Cat>
</ArrayOfCat>

Есть ли способ получить один и тот же Root все время при получении этих Сущностей?

Пример:

<Entity>
  <Cat>
    <Name>Tom</Name>
    <Age>2</Age>
  </Cat>
  <Cat>
    <Name>Bob</Name>
    <Age>3</Age>
  </Cat>
</Entity>

Также обратите внимание, что я не собираюсь десериализовать XML обратно в List<Cat>


person Matthew Grima    schedule 06.08.2009    source источник
comment
Что вы подразумеваете под получением одного и того же рута все время? Пожалуйста, дайте более подробную информацию...   -  person Thomas Levesque    schedule 06.08.2009


Ответы (4)


Есть очень простой способ:

public XmlDocument GetEntityXml<T>()
{
    XmlDocument xmlDoc = new XmlDocument();
    XPathNavigator nav = xmlDoc.CreateNavigator();
    using (XmlWriter writer = nav.AppendChild())
    {
        XmlSerializer ser = new XmlSerializer(typeof(List<T>), new XmlRootAttribute("TheRootElementName"));
        ser.Serialize(writer, parameters);
    }
    return xmlDoc;
}
person Suminder    schedule 26.08.2009
comment
При таком подходе XmlSerializer необходимо статически кэшировать и повторно использовать, чтобы избежать серьезной утечки памяти, см. раздел утечка памяти с использованием StreamReader и XmlSerializer. для деталей. - person dbc; 02.09.2018

Если я правильно понимаю, вы хотите, чтобы корень документа всегда был одним и тем же, независимо от типа элемента в коллекции? В этом случае вы можете использовать XmlAttributeOverrides :

       XmlAttributeOverrides overrides = new XmlAttributeOverrides();
       XmlAttributes attr = new XmlAttributes();
       attr.XmlRoot = new XmlRootAttribute("TheRootElementName");
       overrides.Add(typeof(List<T>), attr);
       XmlSerializer serializer = new XmlSerializer(typeof(List<T>), overrides);
       List<T> parameters = GetAll();
       serializer.Serialize(xmlWriter, parameters);
person Thomas Levesque    schedule 06.08.2009

Лучший способ сделать то же самое:

public XmlDocument GetEntityXml<T>()
{
    XmlAttributeOverrides overrides = new XmlAttributeOverrides();
    XmlAttributes attr = new XmlAttributes();
    attr.XmlRoot = new XmlRootAttribute("TheRootElementName");
    overrides.Add(typeof(List<T>), attr);

    XmlDocument xmlDoc = new XmlDocument();
    XPathNavigator nav = xmlDoc.CreateNavigator();
    using (XmlWriter writer = nav.AppendChild())
    {
        XmlSerializer ser = new XmlSerializer(typeof(List<T>), overrides);
        List<T> parameters = GetAll<T>();
        ser.Serialize(writer, parameters);
    }
    return xmlDoc;
}
person John Saunders    schedule 09.08.2009
comment
Главное, что он сериализуется прямо в XmlDocument. Ваш код требовал синтаксического анализа результатов, чтобы вернуть их обратно в документ. В вашем коде также использовался XmlTextWriter, который в значительной степени устарел. - person John Saunders; 10.08.2009
comment
Это было действительно полезно, но поскольку я использую словарь, который нельзя сериализовать, мне очень помог этот пост: theburningmonk.com/2010/05/ - person Mike Malter; 20.11.2012

так просто....

public static XElement ToXML<T>(this IList<T> lstToConvert, Func<T, bool> filter, string rootName)
{
    var lstConvert = (filter == null) ? lstToConvert : lstToConvert.Where(filter);
    return new XElement(rootName,
       (from node in lstConvert
       select new XElement(typeof(T).ToString(),
       from subnode in node.GetType().GetProperties()
       select new XElement(subnode.Name, subnode.GetValue(node, null)))));

}
person abdul kader jeelani    schedule 27.09.2010