繁体   English   中英

MongoDB驱动程序无法在通过ID过滤期间将自定义序列化程序转换为IBsonSerializer

[英]MongoDB driver unable cast custom serializer to IBsonSerializer during filter by Id

我正在使用C#开发应用程序。 我在从MongoDB获取Id的文档时遇到问题。

我的ID不是ObjectId,而是Guid。

模型是简单的POCO。

public class Folder
{
        public Guid Id { get; set; }
        public int OrgId { get; set; }
        // rest props is not important
}

在创建存储库实例(静态构造函数)的过程中,我注册了类映射:

    static MongoDBFolderRepository()
    {
        Init();
    }

    private static void Init()
    {
        var camelCaseConvention = new ConventionPack { new CamelCaseElementNameConvention() };
        ConventionRegistry.Register("CamelCase", camelCaseConvention, type => true);

        BsonClassMap.RegisterClassMap<Folder>(cm =>
        {
            cm.AutoMap();
            cm.MapIdMember(c => c.Id).SetSerializer(new MyGuidSerializer());
            cm.SetIgnoreExtraElements(true);
        });
    }

而且我还指定了自己的序列化器:

[BsonSerializer(typeof(MyGuidSerializer))]
public class MyGuidSerializer : IBsonSerializer
{
    public object Deserialize(BsonDeserializationContext context, BsonDeserializationArgs args)
    {
        return Guid.Parse(BsonSerializer.Deserialize<string>(context.Reader));
    }

    public void Serialize(BsonSerializationContext context, BsonSerializationArgs args, object value)
    {
        BsonSerializer.Serialize(context.Writer, value.ToString());
    }

    public Type ValueType
    {
        get { return typeof(Guid); }
    }
}

现在,这是引发异常的方法:

public async Task<Folder> GetFolderAsync(Guid folderId)
{
    var database = MongoDBHelper.GetDatabase();
    var collection = database.GetCollection<Folder>(CollectionName);
    return await collection.Find(Builders<Folder>.Filter.Eq(x => x.Id, folderId)).SingleAsync();
}

例外是:

无法将类型为“ MyApp.Repositories.Reporting.Concrete.MyGuidSerializer”的对象转换为类型为“ MongoDB.Bson.Serialization.IBsonSerializer`1 [System.Guid]”的对象。

System.InvalidCastException

通常,序列化是可行的。 如果我将方法更改为此版本:

    public async Task<Folder> GetFolderAsync(Guid folderId)
    {
        var database = MongoDBHelper.GetDatabase();
        var collection = database.GetCollection<Folder>(CollectionName);
        return await collection.Find(Builders<Folder>.Filter.Eq("_id", folderId.ToString())).SingleAsync();
    }

我将得到正确填充的Folder模型作为回报。

我做错了什么 ?

为什么首先要尝试序列化为string

您只需通过直接对Guid进行序列化/反序列化就可以解决您的问题:

public void Serialize(BsonSerializationContext context, BsonSerializationArgs args, Guid value)
    {
        var data = new BsonBinaryData(value);
        context.Writer.WriteBinaryData(data);
    }

public object Deserialize(BsonDeserializationContext context, BsonDeserializationArgs args)
    {
        return BsonSerializer.Deserialize<Guid>(context.Reader);
    }

我不知道为什么,但是在更改IBsonSerializer<Guid>接口后,所有方法都可以使用。

[BsonSerializer(typeof(MyGuidSerializer))]
public class MyGuidSerializer : IBsonSerializer<Guid>
{
    public object Deserialize(BsonDeserializationContext context, BsonDeserializationArgs args)
    {
        return Guid.Parse(BsonSerializer.Deserialize<string>(context.Reader));
    }

    public void Serialize(BsonSerializationContext context, BsonSerializationArgs args, Guid value)
    {
        BsonSerializer.Serialize(context.Writer, value.ToString());
    }

    Guid IBsonSerializer<Guid>.Deserialize(BsonDeserializationContext context, BsonDeserializationArgs args)
    {
        return Guid.Parse(BsonSerializer.Deserialize<string>(context.Reader));
    }

    public void Serialize(BsonSerializationContext context, BsonSerializationArgs args, object value)
    {
        BsonSerializer.Serialize(context.Writer, value.ToString());
    }

    public Type ValueType
    {
        get { return typeof(Guid); }
    }
}

一般通用接口继承IBsonSerializer,所以我不明白为什么,如果我实现IBsonSerializer为MyGuidSerializer我Unable cast to IBsonSerializer

是的,但是将实现的接口从IBsonSerializer更改为IBsonSerializer<Guid>解决问题。

Mongodb.CsharpDriver 2.4.4

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM