在 BigCache 中存储任意类型(interface{})

alandtsang · · 1608 次点击 · · 开始浏览    
这是一个创建于 的文章,其中的信息可能已经有所发展或是发生改变。

这篇文章也发在我的个人 [博客](https://calebschoepp.com/blog) 最近在工作中,我的任务是向我们的一个 Golang 服务添加缓存。这个服务需要传入请求以提供用于身份验证的 API key。因此,对于每个请求,该服务都会额外查询数据库以验证 API key,尽管它通常是相同的 key。这很不好。实现缓存最终比我想象的要难得多。 经过调研和工程师之间详尽讨论之后,我们认为 [BigCache](https://github.com/allegro/bigcache) 最适合我们的需求。 这里有一个问题。BigCache 中的 set 方法的声明为 `Set(key string, entry []byte) error`。它期望存储一个 byte slice。但是我们希望存储一个 struct,该 struct 具有多个表示 API key 的字段。这次我们可能只能够存储实际 key 的 bytes。但这只是推迟解决问题。 我们需要的是像其他 Golang 缓存实现中的声明 `Set(key, entry interface{})`。这样我们就可以存储任何我们想要的东西。 这个问题的明显解决方案是序列化。如果我们可以将任意结构序列化为 byte slice,那么我们可以存储任何内容。要使用我们存储的结构,可以从缓存中反序列化获取 byte slice。序列化结构就像在 Golang 中导入任意数量的可用 encoding 库一样容易。但是现在头疼的问题来了。当我们反序列化 bytes 时,Go 语言如何知道将数据存入什么类型的结构?事实证明,Golang 特有的序列化库 `encoding/gob` 具有此功能。 我强烈建议您阅读 Rob Pike 写的关于 Gob 的 [博客文章](https://blog.golang.org/gob),这是一篇好文章。简而言之,Gob 是一种 Go 原生的数据序列化方式,它还具有序列化 interface 类型的功能。为此,您需要在序列化之前使用恰当命名的 [register funtion](https://golang.org/pkg/encoding/gob/#Register) 注册您的类型。我在这里卡住了,因为我找到的关于 `register()` 的任何代码示例总是注册一个单一的 struct 或 interface;我需要注册任意 `interface{}` 类型。我在 Go playground 上摸索了一下,发现它也可以做到。 ```go // 大多数示例中注册的类型 type foo struct { bar string } gob.register(foo{}) // 我想要注册的类型 var type interface{} // 可以是任何结构 gob.register(type) ``` ## 把它们组合在一起 解决了将任意 struct 存储为 bytes 的问题后,我将向您展示如何将它们组合在一起。首先,我们需要一个缓存 interface,以便系统的其余部分能够与之交互。对于一个简单的缓存,我们只需要 get 和 set 方法。 ```go type Cache interface { Set(key, value interface{}) error Get(key interface{}) (interface{}, error) } ``` 现在,让我们定义实现上述接口的 BigCache 实现。首先,我们需要一个结构来保存缓存并可以向其中添加方法。 您还可以在此结构中添加其他字段,例如 metrics。 ```go type bigCache struct { cache *bigcache.BigCache } ``` 接下来是 get 和 set 方法的实现。两种方法都断言 key 是 string。由此开始,get 和 set 的实现就彼此独立了。一个序列化一个值并存储它。另一个获取值并将其反序列化。 ```go func (c *bigCache) Set(key, value interface{}) error { // 断言 key 为 string 类型 keyString, ok := key.(string) if !ok { return errors.New("a cache key must be a string") } // 将 value 序列化为 bytes valueBytes, err := serialize(value) if err != nil { return err } return c.cache.Set(keyString, valueBytes) } func (c *bigCache) Get(key interface{}) (interface{}, error) { // 断言 key 为 string 类型 keyString, ok := key.(string) if !ok { return nil, errors.New("a cache key must be a string") } // 获取以 bytes 格式存储的 value valueBytes, err := c.cache.Get(keyString) if err != nil { return nil, err } // 反序列化 valueBytes value, err := deserialize(valueBytes) if err != nil { return nil, err } return value, nil } ``` 最后是 `encoding/gob` 序列化逻辑。除了使用 `register()` 之外,这是 Go 中序列化内容相当标准的用法。 ```go func serialize(value interface{}) ([]byte, error) { buf := bytes.Buffer{} enc := gob.NewEncoder(&buf) gob.Register(value) err := enc.Encode(&value) if err != nil { return nil, err } return buf.Bytes(), nil } func deserialize(valueBytes []byte) (interface{}, error) { var value interface{} buf := bytes.NewBuffer(valueBytes) dec := gob.NewDecoder(buf) err := dec.Decode(&value) if err != nil { return nil, err } return value, nil } ``` 通过这些,我们已经设法在 BigCache 中存储 `interface{}` 值了。现在我的团队的服务效率提高了一些。太酷了!如果您正在寻找一个更全面的实现,请查看我的 [gist](https://gist.github.com/calebschoepp/0165d92de412e288aa7441e792d0aa3a)。 如果您喜欢这篇文章,请查看我的 [博客](https://calebschoepp.com/blog) 以获取类似内容。

via: https://dev.to/calebschoepp/storing-empty-interfaces-in-bigcache-1b33

作者:calebschoepp  译者:alandtsang  校对:polaris1119

本文由 GCTT 原创编译,Go语言中文网 荣誉推出


有疑问加站长微信联系(非本文作者))

入群交流(和以上内容无关):加入Go大咖交流群,或添加微信:liuxiaoyan-s 备注:入群;或加QQ群:692541889

1608 次点击  ∙  1 赞  
加入收藏 微博
被以下专栏收入,发现更多相似内容
暂无回复
添加一条新回复 (您需要 登录 后才能回复 没有账号 ?)
  • 请尽量让自己的回复能够对别人有帮助
  • 支持 Markdown 格式, **粗体**、~~删除线~~、`单行代码`
  • 支持 @ 本站用户;支持表情(输入 : 提示),见 Emoji cheat sheet
  • 图片支持拖拽、截图粘贴等方式上传