我正在使用 Couchbase 来存储和管理我的数据。
当我尝试使用缓存从缓存中获取实体时
CacheClient.GetWithCas<T>(cacheKey)
我遇到了这个错误:
Unable to load type X required for deserialization.
at System.Runtime.Serialization.ObjectManager.DoFixups()
at System.Runtime.Serialization.Formatters.Binary.ObjectReader.Deserialize(HeaderHandler handler, __BinaryParser serParser, Boolean fCheck, Boolean isCrossAppDomain, IMethodCallMessage methodCallMessage)
at System.Runtime.Serialization.Formatters.Binary.BinaryFormatter.Deserialize(Stream serializationStream, HeaderHandler handler, Boolean fCheck, Boolean isCrossAppDomain, IMethodCallMessage methodCallMessage)
at System.Runtime.Serialization.Formatters.Binary.BinaryFormatter.Deserialize(Stream serializationStream, HeaderHandler handler, Boolean fCheck, IMethodCallMessage methodCallMessage)
at Enyim.Caching.Memcached.DefaultTranscoder.DeserializeObject(ArraySegment`1 value)
at Enyim.Caching.Memcached.DefaultTranscoder.Deserialize(CacheItem item)
at Enyim.Caching.Memcached.DefaultTranscoder.Enyim.Caching.Memcached.ITranscoder.Deserialize(CacheItem item)
at Couchbase.CouchbaseClient.PerformTryGet(String key, UInt64& cas, Object& value)
at Enyim.Caching.MemcachedClient.GetWithCas[T](String key)
我想这可能是因为我更改了类型(删除属性或更改其签名)。
你能帮我弄清楚如何避免这种情况发生吗?
最佳答案
如果你只想反序列化最新版本的类型,你可以通过两种方式进行:
- 修改 memcached 转码器并使其能够处理异常(硬方式)
- 甚至不要尝试反序列化旧类型(简单方法)
我将给出最后一种方式的例子。首先,您需要稍微修改所有缓存键。您需要向其中添加“命名空间”:之前 - mycachekey:myparam
,之后 - mychachekey:cachekeynamespace:myparam
。
那么什么是“命名空间”呢? - “命名空间”是简单的计数器,您需要在每次类型更改后递增。您也可以将它存储在 couchbase/memcached 中并使用 increment
函数来更改它的值。通过这种方式,您永远不会从缓存中获取过时的值。关于该命名空间如何工作的简短示例:
- 获取 key 的命名空间,如果不存在则创建它到 var
ckns
; - 创建缓存键
foo
(这里 foo 也将代表类型的名称) 使用参数1
:cacheKey = 'foo' + ':' + ckns + ':' + 1
; - 使用键存储值“bar”:
foo:0:1
; - 更改类型及其命名空间版本;
- 尝试使用参数
1
从缓存键foo
中获取(但请记住 命名空间已更改); - 你会得到 key:
foo:1:1
结果你会得到 chache miss,也不异常(exception),因为反序列化没什么。
还有一些注意事项:
- 如果您只需要缓存,最好使用 memcached(包含在内 在 couchbase 安装中)比 couchbase。
- 如果您想在序列化方面获得更好的性能并 减少存储在缓存中的数据量看看不同的 序列化“引擎”。好的选择可以序列化为 JSON(文本 序列化)或使用 Protobuf(二进制序列化,非常紧凑)*
*还有用于 protobuf 序列化的 memcached 转码器,可以找到 here .但是要将它与自定义类型一起使用,您需要应用 here 中描述的一点修复.
关于c# - 无法加载反序列化所需的类型,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/18592886/