Почему мой DBNull не является синглтоном, когда я десериализирую его с помощью XmlSerialiser?
-
09-09-2019 - |
Вопрос
Я всегда предполагал, что DBNull.value был синглтоном.И таким образом, вы могли бы делать подобные вещи:
VB.NET:
If someObject Is DbNull.Value Then
...
End if
C#:
If (someObject == DbNull.Value)
{
...
}
Но недавно я сериализовал экземпляр DBNull с помощью XmlSerialiser, и внезапно это больше не был синглтон.Операции сравнения типов (например, C # (obj - это 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.
Решение
Хотя DBNull.Value
является static readonly
и существует только как единственный экземпляр...когда вы десериализуете, код сериализации будет создавать новый экземпляр класса DBNull
из "данных" в потоке.С тех пор как DBNull.Value
является просто DBNull
например, у сериализации нет способа узнать, что это "особый" экземпляр.
ПРИМЕЧАНИЕ:
По той же причине, если вы создадите свой собственный класс с экземпляром 'singleton', который вы сериализуете, а затем десериализуете, вы получите точно такое же поведение.Хотя десериализованный экземпляр будет неотличим от исходного экземпляра, они не будут тот же самый экземпляр.
Другие советы
Ваш код на c # не соответствует вызову .Метод Equals .Не протестировав это, я на самом деле почти уверен, что если вы заменили
someObject == DbNull.Value
с
DbNull.Value.Equals(someObject)
это дало бы вам ожидаемый результат.Для получения некоторых сведений о операторе равенства и методе Equals взгляните на:Запись в блоге Эрика Липпертса на эту тему