DataContract Сериализация XML и атрибуты XML

Можно ли десериализовать этот XML-объект в объект, помеченный атрибутом DataContract?

<root>
<distance units="m">1000</distance>
</root>

Как вы видите, есть атрибут "units". Я не считаю, что это поддерживается. Или я не прав?

Ответ 1

Это может быть достигнуто, но вам придется переопределить сериализатор по умолчанию, применив атрибут [XmlSerializerFormat] к DataContract. Хотя это можно сделать, это не работает так же, как сериализатор по умолчанию, поэтому используйте его с осторожностью.

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

using ...
using System.Runtime.Serialization;
using System.ServiceModel;
using System.Xml.Serialization;

[DataContract]
[XmlSerializerFormat]
public class root
{
   public distance distance=new distance();
}

[DataContract]
public class distance
{
  [DataMember, XmlAttribute]
  public string units="m";

  [DataMember, XmlText]
  public int value=1000;
}

Вы можете проверить это с помощью следующего кода:

root mc = new root();
XmlSerializer ser = new XmlSerializer(typeof(root));
StringWriter sw = new StringWriter();
ser.Serialize(sw, mc);
Console.WriteLine(sw.ToString());
Console.ReadKey();

Выход будет:

<?xml version="1.0" encoding="utf-16"?>
<root xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
                                   xmlns:xsd="http://www.w3.org/2001/XMLSchema">
  <distance units="m">1000</distance>
</root>

Ответ 2

Сериализатор данных, используемый по умолчанию в WCF, не поддерживает атрибуты XML по причинам производительности (DCS примерно на 10% быстрее, чем XML-сериализатор).

Итак, если вы действительно хотите использовать DCS, вы не можете использовать эту структуру, которую у вас есть - ее нужно было бы изменить.

Или вам нужно использовать XmlSerializer с WCF, как показал Грег в его ответе, - это тоже работает, но вы теряете преимущество в производительности (плюс все другие преимущества) DCS.