Порядок сериализации .NET - PullRequest
14 голосов
/ 19 июня 2009

Я пытаюсь сериализовать некоторые объекты, используя XmlSerializer и наследование, но у меня возникают некоторые проблемы с упорядочением результата.

Ниже приведен пример, аналогичный тому, который я настроил: ~

public class SerializableBase
{
    [XmlElement(Order = 1)]
    public bool Property1 { get; set;}

    [XmlElement(Order = 3)]
    public bool Property3 { get; set;}
}

[XmlRoot("Object")]
public class SerializableObject1 : SerializableBase
{
}

[XmlRoot("Object")]
public class SerializableObject2 : SerializableBase
{
    [XmlElement(Order = 2)]
    public bool Property2 { get; set;}
}

Результат, который я хочу, следующий: ~

<Object>
    <Property1></Property1>
    <Property2></Property2>
    <Property3></Property3>
</Object>

Однако я получаю результат: ~

<Object>
    <Property1></Property1>
    <Property3></Property3>
    <Property2></Property2>
</Object>

Кто-нибудь знает, возможно ли это или какая-либо альтернатива?

Спасибо

Ответы [ 6 ]

17 голосов
/ 21 июня 2009

Технически, с точки зрения чистого XML, я бы сказал, что, вероятно, это плохая вещь.

.NET скрывает большую часть сложности таких вещей, как XmlSerialization - в этом случае она скрывает схему, которой должен соответствовать ваш сериализованный XML.

Выводимая схема будет использовать элементы последовательности для описания базового типа и типов расширений. Это требует строгого упорядочения - даже если десериализатор менее строг и принимает неприспособленные элементы.

В XML-схемах при определении типов расширений дополнительные элементы из дочернего класса должны идти после элементов из базового класса.

по сути, у вас есть схема, которая выглядит примерно так (теги xml-y удалены для ясности)

base
  sequence
    prop1
    prop3

derived1 extends base
  sequence
    <empty>

derived2 extends base
  sequence
    prop2

Нет способа вставить заполнитель между prop1 и prop3, чтобы указать, куда могут перейти свойства из производного xml.

В конце концов, у вас есть несоответствие между вашим форматом данных и вашим бизнес-объектом. Вероятно, ваша лучшая альтернатива - определить объект для вашей сериализации xml.

Например

[XmlRoot("Object")
public class SerializableObjectForPersistance
{
    [XmlElement(Order = 1)]
    public bool Property1 { get; set; }

    [XmlElement(Order = 2, IsNullable=true)]
    public bool Property2 { get; set; }

    [XmlElement(Order = 3)]
    public bool Property3 { get; set; }
}

Это отделяет ваш код сериализации xml от вашей объектной модели. Скопируйте все значения из SerializableObject1 или SerializableObject2 в SerializableObjectForPersistance, а затем сериализуйте его.

По сути, если вам нужен такой специфический контроль над форматом сериализованного XML, который не совсем соответствует структуре сериализации ожидаемого XML, вам необходимо отделить дизайн бизнес-объекта (в данном случае структуру наследования) и ответственность за сериализация этого бизнес-объекта.

4 голосов
/ 20 июня 2009

РЕДАКТИРОВАТЬ: Этот подход не работает . Я оставил пост, чтобы люди могли избежать такого мышления.

Сериализатор действует рекурсивно. В этом есть польза; при десериализации процесс десериализации может считывать базовый класс, а затем производный класс. Это означает, что свойство в производном классе не устанавливается перед свойствами в базе, что может привести к проблемам.

Если это действительно имеет значение (и я не уверен, почему так важно привести их в порядок), то вы можете попробовать это -

1) сделать базовый класс Property1 и Property3 виртуальным. 2) переопределите их с тривиальными свойствами в вашем производном классе. Например,

public class SerializableBase
{
    [XmlElement(Order = 1)]
    public virtual bool Property1 { get; set;}

    [XmlElement(Order = 3)]
    public virtual bool Property3 { get; set;}
}

[XmlRoot("Object")]
public class SerializableObject1 : SerializableBase
{
}

[XmlRoot("Object")]
public class SerializableObject2 : SerializableBase
{
    [XmlElement(Order = 1)]
    public override bool Property1 
    { 
      get { return base.Property1; }
      set { base.Property1 = value; }
    }

    [XmlElement(Order = 2)]
    public bool Property2 { get; set;}

    [XmlElement(Order = 3)]
    public override bool Property3
    { 
      get { return base.Property3; }
      set { base.Property3 = value; }
    }

}

Это дает конкретную реализацию свойства наиболее производному классу, и порядок должен соблюдаться.

3 голосов
/ 10 сентября 2013

Этот пост довольно старый, но у меня недавно была похожая проблема в WCF, и я нашел решение, похожее на описанное выше Стивом Купером, но такое, которое действительно работает, и предположительно будет работать и для сериализации XML.

Если вы удалите атрибуты XmlElement из базового класса и добавите копию каждого свойства с другим именем в производные классы, которые получают доступ к базовому значению через get / set, копии могут быть сериализованы с соответствующим назначенным именем используя XmlElementAttribute, и, надеюсь, будет сериализован в порядке по умолчанию:

public class SerializableBase
{
   public bool Property1 { get; set;}
   public bool Property3 { get; set;}
}

[XmlRoot("Object")]
public class SerializableObject : SerializableBase
{
  [XmlElement("Property1")]
  public bool copyOfProperty1 
  { 
    get { return base.Property1; }
    set { base.Property1 = value; }
  }

  [XmlElement]
  public bool Property2 { get; set;}

  [XmlElement("Property3")]
  public bool copyOfProperty3
  { 
    get { return base.Property3; }
    set { base.Property3 = value; }
  }
}

Я также добавил интерфейс для добавления к производным классам, чтобы сделать копии обязательными:

interface ISerializableObjectEnsureProperties
{
  bool copyOfProperty1  { get; set; }
  bool copyOfProperty2  { get; set; }
}

Это не существенно, но означает, что я могу проверить, все ли реализовано во время компиляции, а не проверять результирующий XML. Первоначально я создал эти абстрактные свойства SerializableBase, но затем они сначала сериализуются (с базовым классом), что, как я теперь понимаю, логично.

Это вызывается обычным способом, изменяя одну строку выше:

public class SerializableObject : SerializableBase, ISerializableObjectEnsureProperties

Я только протестировал это в WCF и перенес концепцию в XML-сериализацию без компиляции, поэтому, если это не сработает, извините, но я ожидаю, что он будет вести себя таким же образом - уверен, кто-то дайте мне знать, если нет ...

3 голосов
/ 19 июня 2009

Похоже, класс XmlSerializer сериализует базовый тип, а затем производные типы в этом порядке и учитывает только свойство Order внутри каждого класса в отдельности. Даже если заказ не совсем то, что вы хотите, он все равно должен десериализоваться должным образом. Если вам действительно нужен такой порядок, вам нужно написать собственный сериализатор xml. Я хотел бы предостеречь против этого, потому что .NET XmlSerializer делает много специальной обработки для вас. Можете ли вы описать, почему вам нужны вещи в указанном вами порядке?

2 голосов
/ 28 мая 2014

Я знаю, что этот вопрос истек; однако вот решение этой проблемы:

Имя метода всегда должно начинаться с ShouldSerialize, а затем заканчиваться именем свойства. Затем вам просто нужно вернуть логическое значение, основанное на любых условных выражениях, относительно того, сериализовать ли значение или нет.

public class SerializableBase
{
    public bool Property1 { get; set;}
    public bool Property2 { get; set;}
    public bool Property3 { get; set;}

    public virtual bool ShouldSerializeProperty2 { get { return false; } }
}

[XmlRoot("Object")]
public class SerializableObject1 : SerializableBase
{        
}

[XmlRoot("Object")]
public class SerializableObject2 : SerializableBase
{
    public override bool ShouldSerializeProperty2 { get { return true; } }
}

Результат при использовании SerializableObject2: ~

<Object>
    <Property1></Property1>
    <Property2></Property2>
    <Property3></Property3>
</Object>

Результат при использовании SerializableObject1: ~

<Object>
    <Property1></Property1>
    <Property3></Property3>
</Object>

Надеюсь, это поможет многим другим!

0 голосов
/ 05 сентября 2012

Как сказал Надер, возможно, стоит подумать о создании более слабосвязанной конструкции. Однако, в моем случае, слабая связь не была подходящей. Вот моя иерархия классов и то, как я предлагаю решить проблему без использования настраиваемой сериализации или DTO.

В моем проекте я создаю целую кучу объектов для представления фрагментов XML-документа, который будет представлен через веб-сервис. Есть очень большое количество штук. Не все отправляются с каждым запросом (на самом деле, в этом примере я моделирую ответ, но концепции совпадают). Эти части используются во многом как строительные блоки для сборки запроса (или дизассемблирования ответа, в данном случае). Итак, вот пример использования агрегации / инкапсуляции для выполнения желаемого упорядочения, несмотря на иерархию наследования.

[Serializable]
public abstract class ElementBase
{
    // This constructor sets up the default namespace for all of my objects. Every
    // Xml Element class will inherit from this class.
    internal ElementBase()
    {
        this._namespaces = new XmlSerializerNamespaces(new XmlQualifiedName[] {
            new XmlQualifiedName(string.Empty, "urn:my-default-namespace:XSD:1")
        });
    }

    [XmlNamespacesDeclaration]
    public XmlSerializerNamespaces Namespaces { get { return this._namespaces; } }
    private XmlSerializationNamespaces _namespaces;
}


[Serializable]
public abstract class ServiceBase : ElementBase
{
    private ServiceBase() { }

    public ServiceBase(Guid requestId, Guid? asyncRequestId = null, Identifier name = null)
    {
        this._requestId = requestId;
        this._asyncRequestId = asyncRequestId;
        this._name = name;
    }

    public Guid RequestId
    {
        get { return this._requestId;  }
        set { this._requestId = value;  }
    }
    private Guid _requestId;

    public Guid? AsyncRequestId
    {
        get { return this._asyncRequestId; }
        set { this._asyncRequestId = value; }
    }
    private Guid? _asyncRequestId;

    public bool AsyncRequestIdSpecified
    {
        get { return this._asyncRequestId == null && this._asyncRequestId.HasValue; }
        set { /* XmlSerializer requires both a getter and a setter.*/ ; }
    }

    public Identifier Name
    {
        get { return this._name; }
        set { this._name; }
    }
    private Identifier _name;
}


[Serializable]
public abstract class ServiceResponseBase : ServiceBase
{
    private ServiceBase _serviceBase;

    private ServiceResponseBase() { }

    public ServiceResponseBase(Guid requestId, Guid? asyncRequestId = null, Identifier name = null, Status status = null)
    {
        this._serviceBase = new ServiceBase(requestId, asyncRequestId, name);
        this._status = status;
    }

    public Guid RequestId
    {
        get { return this._serviceBase.RequestId; }
        set { this._serviceBase.RequestId = value; }
    }

    public Guid? AsyncRequestId
    {
        get { return this._serviceBase.AsyncRequestId; }
        set { this._serviceBase.AsyncRequestId = value; }
    }

    public bool AsynceRequestIdSpecified
    {
        get { return this._serviceBase.AsyncRequestIdSpecified; }
        set { ;  }
    }

    public Identifier Name
    {
        get { return this._serviceBase.Name; }
        set { this._serviceBase.Name = value; }
    }

    public Status Status
    {
        get { return this._status; }
        set { this._status = value; }
    }
}

[Serializable]
[XmlRoot(Namespace = "urn:my-default-namespace:XSD:1")]
public class BankServiceResponse : ServiceResponseBase
{
    // Determines if the class is being deserialized.
    private bool _isDeserializing;

    private ServiceResponseBase _serviceResponseBase;

    // Constructor used by XmlSerializer.
    // This is special because I require a non-null List<T> of items later on.
    private BankServiceResponse()
    { 
        this._isDeserializing = true;
        this._serviceResponseBase = new ServiceResponseBase();
    }

    // Constructor used for unit testing
    internal BankServiceResponse(bool isDeserializing = false)
    {
        this._isDeserializing = isDeserializing;
        this._serviceResponseBase = new ServiceResponseBase();
    }

    public BankServiceResponse(Guid requestId, List<BankResponse> responses, Guid? asyncRequestId = null, Identifier name = null, Status status = null)
    {
        if (responses == null || responses.Count == 0)
            throw new ArgumentNullException("The list cannot be null or empty", "responses");

        this._serviceResponseBase = new ServiceResponseBase(requestId, asyncRequestId, name, status);
        this._responses = responses;
    }

    [XmlElement(Order = 1)]
    public Status Status
    {
        get { return this._serviceResponseBase.Status; }
        set { this._serviceResponseBase.Status = value; }
    }

    [XmlElement(Order = 2)]
    public Guid RequestId
    {
        get { return this._serviceResponseBase.RequestId; }
        set { this._serviceResponseBase.RequestId = value; }
    }

    [XmlElement(Order = 3)]
    public Guid? AsyncRequestId
    {
        get { return this._serviceResponseBase.AsyncRequestId; }
        set { this._serviceResponseBase.AsyncRequestId = value; }
    }

    [XmlIgnore]
    public bool AsyncRequestIdSpecified
    {
        get { return this._serviceResponseBase.AsyncRequestIdSpecified; }
        set { ; } // Must have this for XmlSerializer.
    }

    [XmlElement(Order = 4)]
    public Identifer Name
    {
         get { return this._serviceResponseBase.Name; }
         set { this._serviceResponseBase.Name; }
    }

    [XmlElement(Order = 5)]
    public List<BankResponse> Responses
    {
        get { return this._responses; }
        set
        {
            if (this._isDeserializing && this._responses != null && this._responses.Count > 0)
                this._isDeserializing = false;

            if (!this._isDeserializing && (value == null || value.Count == 0))
                throw new ArgumentNullException("List cannot be null or empty.", "value");

            this._responses = value;
        }
    }
    private List<BankResponse> _responses;
}

Таким образом, хотя мне нужно создать свойства для всех содержащихся классов, я могу делегировать любую собственную логику, которую я мог бы иметь в установщиках / получателях свойств содержащихся классов (классов), просто используя свойства вложенного класса, когда свойства конечного класса Доступ Поскольку наследования нет, я могу украсить все свойства класса листа с помощью атрибута XmlElementAttribute и использовать любой порядок, который я считаю подходящим.


UPDATE:

Я вернулся, чтобы вернуться к этой статье, потому что мои дизайнерские решения об использовании наследования классов вернулись, чтобы снова укусить меня. Хотя мое решение, приведенное выше, работает, я его использую, но я действительно считаю, что решение Nader является лучшим и должно быть рассмотрено до того, как решение, которое я представил. На самом деле, я сегодня +1 к нему! Мне очень нравится его ответ, и если мне когда-нибудь удастся реорганизовать мой текущий проект, я определенно отделю бизнес-объект от логики сериализации для объектов, которые иначе получили бы большую выгоду от наследования, чтобы упростить код и упростить его. для других, чтобы использовать и понимать.

Спасибо за публикацию вашего ответа, Nader, так как я думаю, что многие найдут его очень поучительным и полезным.

...