前言
终于结束了两个月的找工作战役,前前后后面试了几十场,结果不是很理想,但是也还不错。经过这次换工作,我的感慨颇多,深刻感到自己最近2年多过于懈怠,成长缓慢。痛定思痛,今后一定要在一个领域深耕下去,提升自己的专业能力。Golang是我目前和接下来一段时间内的主要工作语言,遂决定在Golang的领域做深,做强,成为Golang专家,矢志不渝。
今天在Go中文社区看到翻译小组,个人一直对翻译英文文档颇感兴趣,因此决定翻译一篇文章。
原文地址:
Storing Empty Interfaces in BigCache
https://dev.to/calebschoepp/storing-empty-interfaces-in-bigcache-1b33
译文
本文也发表在我的个人博客
博客
https://calebschoepp.com/blog
在近期的工作中,我被安排给我们的一个Golang服务加缓存。这个特殊的服务的功能是给新来的请求提供一个接口鉴权用的键。所以对于每一个请求,即使使用相同的键,这个服务也产生了一次额外的数据库请求来验证键的合法性。这可不太妙。实现这个缓存最后比我刚开始预想的要更困难。
在做了一些调研和工程师们的详细的讨论后,我们得出了BigCache(大缓存)最适合我们的需要的结论。
这是个陷阱。在BigCache中的set方法的格式是Set(key string, entry []byte) error。这个方法希望使用者存储一个字节切片。不过,我们想要存储的是一个用多个字段来表示API接口键的结构体。这一次我们可以只存储实际的键的字节。但是这只不过是延迟解决这个问题而已。我们需要像其他Golang缓存实现的定义,Set(key, entry interface{})。这样我们就可以存储我们想存的任何类型了。
这个问题最明显的解决方案是序列化。如果我们能把一个实体结构体序列化成一个字节切片,那么我们就可以存储任何东西了。如果我们想使用我们存储的结构体元素,我们可以从缓存中获取字节切片,然后反序列化。序列化结构体既重要又简单,在Golang中有一些可用的编码库供使用。不过,现在让人头疼的问题来了。当我们反序列化字节时,语言本身如何才能知道结构体有哪些字段,并且如何把对应的数据填充进去呢?最后发现了encoding/gob
, 这是Golang 特有的序列化库,就有这个能力。
强烈推荐你去阅读Rob Pike发表的一篇讨论Gob的博客;那是一个很好的文章。简单来说,Gob是Go语言原生的序列化数据方法,此外他也有序列化接口类型的能力。为了实现这个功能,你需要在序列化你的类型前,用一个合适的命名注册函数来注册你的类型。最后我被卡在这了,因为我发现任何register()的代码示例,都会注册一个具体的结构体或者接口标识;我需要他能注册一个任意的接口类型。尽管有一些混乱,但是我发现在Go语言的原生中也可以实现。
// 大部分代码示例这样做 type foo struct { bar string } gob.register(foo{}) // 我需要的并且发现可行 var type interface{} // 可以是任何类型 gob.register(type)
全部合到一起
由于存储任意的结构体成字节的问题解决了,接下来我就展示我怎么把它们放到一起的。首先我们想要一个缓存使用的接口,系统中其他部分可以相互作用。对于一个简单的缓存系统,我们不需要比get和set更多的方法了。
type Cache interface { Set(key, value interface{}) error Get(key interface{}) (interface{}, error) }
现在我们来定义BigCache的实现,兑现上面的接口设计。首先我们需要一个结构体承载缓存的数据,并且给它配备对应的方法。你也可以在结构体里加入其他的东西,比如一些指标。
type bigCache struct { cache *bigcache.BigCache }
接下来是get和set方法的实现。两个方法都断言键key是字符串类型。这里,get和set方法是互相对照的。一个是序列化数据并且存储数据。另一个是读取数据并且反序列化。
func (c *bigCache) Set(key, value interface{}) error { // 断言key属于字符串类型 keyString, ok := key.(string) if !ok { return errors.New("a cache key must be a string") } // 序列化value成字节 valueBytes, err := serialize(value) if err != nil { return err } return c.cache.Set(keyString, valueBytes) } func (c *bigCache) Get(key interface{}) (interface{}, error) { // 断言key属于字符串类型 keyString, ok := key.(string) if !ok { return nil, errors.New("a cache key must be a string") } // 读取被存在字节序列里的value valueBytes, err := c.cache.Get(keyString) if err != nil { return nil, err } // 反序列化字节切片里的value value, err := deserialize(valueBytes) if err != nil { return nil, err } return value, nil }
最后encoding/gob序列化逻辑。在使用register()之外,在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{}接口值。现在我团队负责的服务效率更高了一些。漂亮!如果你正在寻找一个更详细的实现,不妨看看我的摘要。
如果你喜欢本文,那么请移步我的博客查看其他内容。
写在后面
文章翻译完了,难度不大,比较好理解。其中有些不是很优雅的地方,欢迎大佬们指点。也欢迎喜欢技术的朋友关注我的公众号,交流技术,交流生活与人生感悟。
文章浏览阅读3.4k次,点赞8次,收藏42次。一、什么是内部类?or 内部类的概念内部类是定义在另一个类中的类;下面类TestB是类TestA的内部类。即内部类对象引用了实例化该内部对象的外围类对象。public class TestA{ class TestB {}}二、 为什么需要内部类?or 内部类有什么作用?1、 内部类方法可以访问该类定义所在的作用域中的数据,包括私有数据。2、内部类可以对同一个包中的其他类隐藏起来。3、 当想要定义一个回调函数且不想编写大量代码时,使用匿名内部类比较便捷。三、 内部类的分类成员内部_成员内部类和局部内部类的区别
文章浏览阅读118次。分布式系统要求拆分分布式思想的实质搭配要求分布式系统要求按照某些特定的规则将项目进行拆分。如果将一个项目的所有模板功能都写到一起,当某个模块出现问题时将直接导致整个服务器出现问题。拆分按照业务拆分为不同的服务器,有效的降低系统架构的耦合性在业务拆分的基础上可按照代码层级进行拆分(view、controller、service、pojo)分布式思想的实质分布式思想的实质是为了系统的..._分布式系统运维工具
文章浏览阅读174次。1.数据源准备2.数据处理step1:数据表处理应用函数:①VLOOKUP函数; ② CONCATENATE函数终表:step2:数据透视表统计分析(1) 透视表汇总不同渠道用户数, 金额(2)透视表汇总不同日期购买用户数,金额(3)透视表汇总不同用户购买订单数,金额step3:讲第二步结果可视化, 比如, 柱形图(1)不同渠道用户数, 金额(2)不同日期..._exce l趋势分析数据量
文章浏览阅读3.3k次。堡垒机可以为企业实现服务器、网络设备、数据库、安全设备等的集中管控和安全可靠运行,帮助IT运维人员提高工作效率。通俗来说,就是用来控制哪些人可以登录哪些资产(事先防范和事中控制),以及录像记录登录资产后做了什么事情(事后溯源)。由于堡垒机内部保存着企业所有的设备资产和权限关系,是企业内部信息安全的重要一环。但目前出现的以下问题产生了很大安全隐患:密码设置过于简单,容易被暴力破解;为方便记忆,设置统一的密码,一旦单点被破,极易引发全面危机。在单一的静态密码验证机制下,登录密码是堡垒机安全的唯一_horizon宁盾双因素配置
文章浏览阅读7.7k次,点赞4次,收藏16次。Chrome作为一款挺不错的浏览器,其有着诸多的优良特性,并且支持跨平台。其支持(Windows、Linux、Mac OS X、BSD、Android),在绝大多数情况下,其的安装都很简单,但有时会由于网络原因,无法安装,所以在这里总结下Chrome的安装。Windows下的安装:在线安装:离线安装:Linux下的安装:在线安装:离线安装:..._chrome linux debian离线安装依赖
文章浏览阅读153次。中国发达城市榜单每天都在刷新,但无非是北上广轮流坐庄。北京拥有最顶尖的文化资源,上海是“摩登”的国际化大都市,广州是活力四射的千年商都。GDP和发展潜力是衡量城市的数字指...
文章浏览阅读3.3k次。前言spark在java使用比较少,多是scala的用法,我这里介绍一下我在项目中使用的代码配置详细算法的使用请点击我主页列表查看版本jar版本说明spark3.0.1scala2.12这个版本注意和spark版本对应,只是为了引jar包springboot版本2.3.2.RELEASEmaven<!-- spark --> <dependency> <gro_使用java调用spark注册进去的程序
文章浏览阅读4.8k次。汽车零部件开发工具巨头V公司全套bootloader中UDS协议栈源代码,自己完成底层外设驱动开发后,集成即可使用,代码精简高效,大厂出品有量产保证。:139800617636213023darcy169_uds协议栈 源代码
文章浏览阅读4.6k次,点赞20次,收藏148次。AUTOSAR基础篇之OS(下)前言首先,请问大家几个小小的问题,你清楚:你知道多核OS在什么场景下使用吗?多核系统OS又是如何协同启动或者关闭的呢?AUTOSAR OS存在哪些功能安全等方面的要求呢?多核OS之间的启动关闭与单核相比又存在哪些异同呢?。。。。。。今天,我们来一起探索并回答这些问题。为了便于大家理解,以下是本文的主题大纲:[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-JCXrdI0k-1636287756923)(https://gite_autosar 定义了 5 种多核支持类型
文章浏览阅读2.2k次,点赞6次,收藏14次。原因:自己写的头文件没有被加入到方案的包含目录中去,无法被检索到,也就无法打开。将自己写的头文件都放入header files。然后在VS界面上,右键方案名,点击属性。将自己头文件夹的目录添加进去。_vs2013打不开自己定义的头文件
文章浏览阅读3.3w次,点赞80次,收藏342次。此时,可以将系统中所有用户的 Session 数据全部保存到 Redis 中,用户在提交新的请求后,系统先从Redis 中查找相应的Session 数据,如果存在,则再进行相关操作,否则跳转到登录页面。此时,可以将系统中所有用户的 Session 数据全部保存到 Redis 中,用户在提交新的请求后,系统先从Redis 中查找相应的Session 数据,如果存在,则再进行相关操作,否则跳转到登录页面。当数据量很大时,count 的数量的指定可能会不起作用,Redis 会自动调整每次的遍历数目。_redis命令
文章浏览阅读449次,点赞3次,收藏3次。URP的设计目标是在保持高性能的同时,提供更多的渲染功能和自定义选项。与普通项目相比,会多出Presets文件夹,里面包含着一些设置,包括本色,声音,法线,贴图等设置。全局只有主光源和附加光源,主光源只支持平行光,附加光源数量有限制,主光源和附加光源在一次Pass中可以一起着色。URP:全局只有主光源和附加光源,主光源只支持平行光,附加光源数量有限制,一次Pass可以计算多个光源。可编程渲染管线:渲染策略是可以供程序员定制的,可以定制的有:光照计算和光源,深度测试,摄像机光照烘焙,后期处理策略等等。_urp渲染管线