Как переименовать <ArrayOf>атрибут XML, сгенерированный после сериализации Списка объектов - PullRequest
27 голосов
/ 28 июня 2010

Я сериализирую Список объектов List<TestObject>, и XmlSerializer генерирует атрибут <ArrayOfTestObject>, я хочу переименовать или удалить его.
Можно ли это сделать, создав новый класс, который инкапсулирует Список как поле?

 [XmlRoot("Container")]    
 public class TestObject
 {
     public TestObject() { }                         
     public string Str { get; set; }                         
 }

 List<TestObject> tmpList = new List<TestObject>();

 TestObject TestObj = new TestObject();
 TestObj.Str = "Test";

 TestObject TestObj2 = new TestObject();
 TestObj2.Str = "xcvxc";

 tmpList.Add(TestObj);
 tmpList.Add(TestObj2);


 XmlWriterSettings settings = new XmlWriterSettings();
 settings.OmitXmlDeclaration = true;
 settings.Indent = true;
 XmlSerializer serializer = new XmlSerializer(typeof(List<TestObject>));

 using (XmlWriter writer = XmlWriter.Create(@"C:\test.xml", settings))
 {              
     XmlSerializerNamespaces namespaces = new XmlSerializerNamespaces();
     namespaces.Add(string.Empty, string.Empty);
     serializer.Serialize(writer, tmpList, namespaces);                            
}


<ArrayOfTestObject>
  <TestObject>
    <Str>Test</Str>
  </TestObject>
  <TestObject>
    <Str>xcvxc</Str>
  </TestObject>
</ArrayOfTestObject>

Ответы [ 4 ]

29 голосов
/ 28 июня 2010

Самый надежный способ - объявить внешний класс DTO:

[XmlRoot("myOuterElement")]
public class MyOuterMessage {
    [XmlElement("item")]
    public List<TestObject> Items {get;set;}
}

и сериализовать , что (т.е. поместить ваш список в другой объект).


Вы можете избегать класса-оболочки, но я бы не стал:

class Program
{
    static void Main()
    {
        XmlSerializer ser = new XmlSerializer(typeof(List<Foo>),
             new XmlRootAttribute("Flibble"));
        List<Foo> foos = new List<Foo> {
            new Foo {Bar = "abc"},
            new Foo {Bar = "def"}
        };
        ser.Serialize(Console.Out, foos);
    }
}

public class Foo
{
    public string Bar { get; set; }
}

Проблема в том, что при использовании пользовательских атрибутов вам необходимобудьте очень осторожными для хранения и повторного использования сериализатора, в противном случае вы получите лотов динамических сборок, загруженных в память.Этого можно избежать, если вы просто используете конструктор XmlSerializer(Type), поскольку он автоматически кеширует это.

6 голосов
/ 17 августа 2017

Измените следующую строку с:

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

Кому:

XmlRootAttribute root = new XmlRootAttribute("TestObjects");     

XmlSerializer serializer = new XmlSerializer(typeof(List<TestObject>), root);

Это должно работать.

2 голосов
/ 13 ноября 2018

Вы можете добавить дополнительный параметр в конструктор XmlSerializer, чтобы по существу назвать корневой элемент.

XmlSerializer xsSubmit = new XmlSerializer(typeof(List<DropDownOption>), new XmlRootAttribute("DropDownOptions"));

Это приведет к следующей структуре:

<DropDownOptions xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:xsd="http://www.w3.org/2001/XMLSchema"> 
    <DropDownOption>
      <ID>1</ID>
      <Description>foo</Description>
    </DropDownOption>
    <DropDownOption>
      <ID>2</ID>
      <Description>bar</Description>
    </DropDownOption>
</DropDownOptions>
1 голос
/ 14 июня 2016

Создайте еще один класс, например:

       [XmlRoot("TestObjects")]
       public class TestObjects: List<TestObject>
       {

       }

Затем примените приведенный ниже код при герметизации:

            XmlSerializer serializer = new XmlSerializer(typeof(TestObjects));
            MemoryStream memStream = new MemoryStream();
            serializer.Serialize(memStream, tmpList);
...