为 MongoDb 处理可为空和不可为空类型的自定义序列化程序

And*_*lin 5 c# serialization mongodb mongodb-.net-driver

我需要使用 IBsonSerializer 实现自定义序列化器。

这就是我所做的:

internal class MyCustomDateTimeSerializer : IBsonSerializer
{
   public object Deserialize(BsonDeserializationContext context, 
         BsonDeserializationArgs args)
   {
       // Deserialization logic
   }

   public void Serialize(BsonSerializationContext context, 
         BsonSerializationArgs args, object value)
   {
       // Serialization logic
   }

   public Type ValueType => typeof(DateTime);
}
Run Code Online (Sandbox Code Playgroud)

然后在 BsonSerializerAttribute 中使用它:

[BsonSerializer(typeof(MyCustomDateTimeSerializer))]
Run Code Online (Sandbox Code Playgroud)

我的问题是我想序列化/反序列化 DateTime 和 Nullable DateTime。

我的 CustomSerializer 的 ValueType 设置为 typeof(DateTime),因此我得到如下异常:

序列化程序的值类型为 System.DateTime 且与成员类型 System.Nullable`1[[System.DateTime..

我没有找到这个问题的任何解决方案。当然,我可以为 Nullable DateTime 和 DateTime 创建两个不同的类,但也许还有另一种选择?

Cod*_*ler 5

如果您检查 MongoDB.Bson 库中SetSerializer方法的源代码,您会发现它对成员类型进行了非常简单的检查:

if (serializer.ValueType != _memberType)
{
    var message = string.Format("Value type of serializer is {0} and does not match member type {1}.", serializer.ValueType.FullName, _memberType.FullName);
    throw new ArgumentException(message, "serializer");
}
Run Code Online (Sandbox Code Playgroud)

该代码检查类型相等性,并且没有办法欺骗它并使其认为DateTimeNullable<DateTime>相等。

但是,您可以使用一种解决方案来拥有一个序列化器类并避免重复代码。您可以将序列化器设为泛型类,并强制它仅接受DateTimeNullable<DateTime>作为类型参数。这是一个示例:

internal class MyCustomDateTimeSerializer<TDateTime> : IBsonSerializer
{
    static MyCustomDateTimeSerializer()
    {
        if (typeof(TDateTime) != typeof(DateTime) && typeof(TDateTime) != typeof(DateTime?))
        {
            throw new InvalidOperationException($"MyCustomDateTimeSerializer could be used only with {nameof(DateTime)} or {nameof(Nullable<DateTime>)}");
        }
    }

    public object Deserialize(BsonDeserializationContext context, BsonDeserializationArgs args)
    {
        // Deserialization logic
    }

    public void Serialize(BsonSerializationContext context, BsonSerializationArgs args, object value)
    {
        // Serialization logic
    }

    public Type ValueType => typeof(TDateTime);
}

public class SomeDocument
{
    // ...

    [BsonSerializer(typeof(MyCustomDateTimeSerializer<DateTime>))]
    public DateTime Date1 { get; set; }

    [BsonSerializer(typeof(MyCustomDateTimeSerializer<DateTime?>))]
    public DateTime? Date2 { get; set; }
}
Run Code Online (Sandbox Code Playgroud)