.Net XmlSerializer: десериализовать CDATA как внутренний текст - PullRequest
6 голосов
/ 29 декабря 2008

У меня проблема с десериализацией CDATA с использованием стандартного .Net XmlSerializer.

Обновление : я получаю XML из внешней системы и не могу влиять на его формат, поэтому не могу включить CData в отдельный элемент атрибута.

Сериализация дает это:

<?xml version="1.0" encoding="utf-16"?>
<MyClass xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:xsd="http://www.w3.org/2001/XMLSchema"><![CDATA[Hello, world!]]></MyClass>

Десериализация не восстанавливает объект в исходное состояние.

Вот класс, который сериализуется:

public class MyClass
{
    string _data;

    [XmlIgnore]
    public string Data
    {
        get { return _data; }
        set { _data = value; }
    }

    [XmlAnyElement]
    public XmlCDataSection CData
    {
        get { return new XmlDataDocument().CreateCDataSection(Data); }
        set { Data = value.Value; }
    }
}

Вот тест, который не проходит:

[Test]
public void CData_as_inner_text_test()
{
    MyClass item = new MyClass();

    item.Data = "Hello, world!";

    XmlSerializer serializer = new XmlSerializer(item.GetType());
    string serialized;

    using (StringWriter sw = new StringWriter())
    {
        serializer.Serialize(sw, item);
        serialized = sw.GetStringBuilder().ToString();
    }

    MyClass deserialized;

    using (StringReader sr = new StringReader(serialized))
    {
        deserialized = (MyClass)serializer.Deserialize(sr);
    }

    Assert.AreEqual(item.Data, deserialized.Data); // For some reason, deserialized.Data == null
}

Я нашел здесь ту же проблему, но ответа нет: XmlSerializer, XmlAnyElement и CDATA

Ответы [ 2 ]

9 голосов
/ 29 декабря 2008

Свойство CData заканчивается нулевым, поскольку содержимое раздела CDATA заканчивается в свойстве Data, где оно игнорируется ...

<MyClass><![CDATA[Hello, world!]]></MyClass>

абсолютно эквивалентно:

<MyClass>Hello, world!</MyClass>

Вам не важно, записывает ли внешнее приложение содержимое MyClass как CData или нет. Аналогично, внешнему приложению не важно, как вы его записываете.

IOW, это должно быть все, что вам нужно:

public class MyClass
{
    string _data;

    [XmlText]
    public string Data
    {
        get { return _data; }
        set { _data = value; }
    }
}
0 голосов
/ 01 февраля 2012

Сначала объявите свойство как XmlCDataSection

public XmlCDataSection ProjectXml { get; set; }

в этом случае projectXml - это строка xml

ProjectXml = new XmlDocument().CreateCDataSection(projectXml);

при сериализации сообщения у вас будет хороший формат (уведомление)

<?xml version="1.0" encoding="utf-16"?>
<MessageBase xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:xsd="http://www.w3.org/2001/XMLSchema" xsi:type="Message_ProjectStatusChanged">
  <ID>131</ID>
  <HandlerName>Plugin</HandlerName>
  <NumRetries>0</NumRetries>
  <TriggerXml><![CDATA[<?xml version="1.0" encoding="utf-8"?><TmData xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" Version="9.0.0" Date="2012-01-31T15:46:02.6003105" Format="1" AppVersion="10.2.0" Culture="en-US" UserID="0" UserRole=""><PROJECT></PROJECT></TmData>]]></TriggerXml>
  <MessageCreatedDate>2012-01-31T20:28:52.4843092Z</MessageCreatedDate>
  <MessageStatus>0</MessageStatus>
  <ProjectId>0</ProjectId>
  <UserGUID>8CDF581E44F54E8BAD60A4FAA8418070</UserGUID>
  <ProjectGUID>5E82456F42DC46DEBA07F114F647E969</ProjectGUID>
  <PriorStatus>0</PriorStatus>
  <NewStatus>3</NewStatus>
  <ActionDate>0001-01-01T00:00:00</ActionDate>
</MessageBase>
...