3

我正在使用 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) 

我想这可能是因为我更改了类型(删除属性或更改了它们的签名)。

你能帮我弄清楚如何避免这种情况发生吗?

4

1 回答 1

0

如果您只想反序列化最新版本的类型,您可以通过 2 种方式进行:

  1. 修改 memcached 转码器并使其能够处理异常(困难的方式)
  2. 甚至不尝试反序列化旧类型(简单的方法)

我将举例说明最后一种方式。首先,您需要稍微修改所有缓存键。您需要为其添加“命名空间”:之前 - mycachekey:myparam,之后 - mychachekey:cachekeynamespace:myparam。那么什么是“命名空间”?- “命名空间”是简单的计数器,您需要在每次类型更改后递增。您也可以将它存储在 couchbase/memcached 中并使用increment函数来更改它的值。通过这种方式,您永远不会从缓存中获得过时的值。关于该命名空间如何工作的简短示例:

  1. 获取键的命名空间,如果不存在则创建它到 var 中ckns
  2. 使用参数创建缓存键foo(这里 foo 也代表类型的名称)1:;cacheKey = 'foo' + ':' + ckns + ':' + 1
  3. 用键存储值'bar' foo:0:1:;
  4. 更改类型及其命名空间版本;
  5. 尝试foo使用参数从缓存键中获取1(但请记住名称空间已更改);
  6. 你会得到 key:foo:1:1结果你会得到只是 chache miss,没有例外,因为它没有什么可以反序列化的。

还有一些注意事项:

  • 如果您只需要缓存,则最好使用 memcached(包含在 couchbase 安装中)而不是 couchbase。
  • 如果您想获得更好的序列化性能并减少存储在缓存中的数据量,请查看不同的序列化“引擎”。好的选择可以序列化为 JSON(文本序列化)或使用 Protobuf(二进制序列化,非常紧凑)*

*还有用于 protobuf 序列化的 memcached 转码器,可在此处找到。但是要将它与自定义类型一起使用,您需要应用此处描述的一些修复。

于 2013-09-04T08:52:01.767 回答