Почему мой DbNull не является синглтоном, когда я десериализую его с помощью XmlSerialiser? - PullRequest
6 голосов
/ 04 июня 2009

Я всегда предполагал, что DbNull.value был синглтоном. И, таким образом, вы могли бы делать такие вещи:

VB.NET:

If someObject Is DbNull.Value Then
    ...
End if

C #:

If (someObject == DbNull.Value)
{
    ...
}

Но недавно я сериализовал экземпляр DbNull с использованием XmlSerialiser, и вдруг он больше не был синглтоном. Хотя операции сравнения типов (например, C # (obj is DBNull)) работают нормально.

Код следует:

[Serializable, System.Xml.Serialization.XmlInclude(typeof(DBNull))]
public class SerialiseMe
{
    public SerialiseMe() { }

    public SerialiseMe(object value)
    {
        this.ICanBeDbNull = value;
    }
    public Object ICanBeDbNull { get; set; }
}

public void Foo()
{
    var serialiseDbNull = new SerialiseMe(DBNull.Value);
    var serialiser = new System.Xml.Serialization.XmlSerializer(typeof(SerialiseMe));
    var ms = new System.IO.MemoryStream();
    serialiser.Serialize(ms, serialiseDbNull);
    ms.Seek(0, System.IO.SeekOrigin.Begin);
    var deSerialisedDbNull = (SerialiseMe)serialiser.Deserialize(ms);

    // Is false, WTF!
    var equalsDbNullDeserialised = deSerialisedDbNull.ICanBeDbNull == DBNull.Value;
    // Is false, WTF!
    var refEqualsDbNullDeserialised = object.ReferenceEquals(deSerialisedDbNull.ICanBeDbNull, DBNull.Value);
    // Is true.
    var convertIsDbNullDeserialised = Convert.IsDBNull(deSerialisedDbNull.ICanBeDbNull);
    // Is true.
    var isIsDbNullDeserialised = deSerialisedDbNull.ICanBeDbNull is DBNull;

}

Почему это так? И как это происходит? И может ли это случиться с другими статическими полями?

PS: я знаю, что пример кода VB выполняет сравнение ссылок, а c # вызывает Object.Equals. Оба имеют одинаковое поведение с DBNull. Я обычно работаю с VB.

Ответы [ 2 ]

7 голосов
/ 04 июня 2009

Хотя DBNull.Value является static readonly и существует только как один экземпляр ... при десериализации код сериализации создаст новый экземпляр класса DBNull из «данных» в поток. Поскольку DBNull.Value - это просто экземпляр DBNull, для сериализации нет способа узнать, что это «особый» экземпляр.

Примечание:
По той же причине, если вы создадите свой собственный класс с экземпляром 'singleton', который вы сериализуете, а затем десериализуете, вы получите точно такое же поведение. Хотя десериализованный экземпляр будет неотличим от исходного экземпляра, он не будет тем же экземпляром .

1 голос
/ 04 июня 2009

Ваш код на c # не равен вызову метода .Equals. Без тестирования я на самом деле уверен, что вы заменили

someObject == DbNull.Value

с

DbNull.Value.Equals(someObject) 

это даст вам ожидаемый результат. Для некоторых деталей об операторе равенства и методе Equals взгляните на: Эрик Липпертс в блоге на эту тему

...