.Net XmlSerializer: десериализация CDATA как внутреннего текста

у меня проблема с десериализацией 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 ответов


свойство CData заканчивается null, поскольку содержимое раздела 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; }
    }
}

сначала объявите свойство как 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>