百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 编程网 > 正文

因为不会Redis的scan命令,我被开除了

yuyutoo 2025-03-13 21:50 2 浏览 0 评论

那个深夜,我登上了公司的服务器,在Redis 命令行里敲入 keys* 后,线上开始报警,服务瞬间被卡死,我只能举起双手,焦急地等待几千万key被慢慢扫描,束手无策万念俱灰的时候,我收到了leader的短信:你明天不用来上班了。

虽然上面是我的臆想,事实上很多公司的运维也会禁用这些命令,来防止开发出错。但我在群里依然看到有同学在问“为什么Redis不能用 keys?我觉得挺好的呀”时,为了不让上面的情况发生,我决定写下这篇文章。

如何才能优雅地遍历Redis?作为一种可以称为数据库的组件,这是多么理所因当的要求。终于,Redis在2.8.0版本新增了众望所归的 scan操作,从此再也不用担心敲入了keys*, 然后等着定时炸弹的引爆。

学会使用 scan并不困难,那么问题又来了,它是如何遍历的?当遍历过程中加入了新的key,当遍历过程中发生了扩容,Redis是如何解决的?抱着深入学习的态度,以及为了能够将来在面试官面前谈笑风生,让我们一起来借此探索Redis的设计原理。

引言

开门见山,首先让我们来总结一下 scan的优缺点。

优点:

  • 提供键空间的遍历操作,支持游标,复杂度O(1), 整体遍历一遍只需要O(N)
  • 提供结果模式匹配
  • 支持一次返回的数据条数设置,但仅仅是个hints,有时候返回更多
  • 弱状态,所有状态只需要客户端需要维护一个游标

缺点:

  • 无法提供完整的快照遍历,也就是中间如果有数据修改,可能有些涉及改动的数据遍历不到
  • 每次返回的数据条数不一定,极度依赖内部实现
  • 返回的数据可能有重复,应用层需要能够处理重入逻辑

所以scan是一个能够满足需求,但也不是完美无瑕的命令。

下面来介绍一下原理,scan到底是如何实现的?scan,hscan等命令主要都是借用了通用的scan操作函数:scanGenericCommand 。

scanGenericCommand 函数分为4步:

  • 解析count和match参数.如果没有指定count,默认返回10条数据
  • 开始迭代集合,如果是key保存为ziplist或者intset,则一次性返回所有数据,没有游标(游标值直接返回0).由于Redis设计,只有数据量比较小的时候才会保存为ziplist或者intset,所以此处不会影响性能.游标在保存为hash的时候发挥作用,具体入口函数为dictScan,下文详细描述。
  • 根据match参数过滤返回值,并且如果这个键已经过期也会直接过滤掉(Redis中键过期之后并不会立即删除)

当迭代一个哈希表时,存在三种情况

  1. 从迭代开始到结束,哈希表没有进行rehash
  2. 从迭代开始到结束,哈希表进行了rehash,但是每次迭代时,哈希表要么没开始rehash,要么已经结束了rehash
  3. 从迭代开始到结束,某次或某几次迭代时哈希表正在进行rehash

在这三种情况之下,sacn是如何实现的?

首先需要知道的前提是:Redis中进行rehash扩容时会存在两个哈希表,ht[0]与ht[1],rehash是渐进式的,即不会一次性完成。新的键值对会存放到ht[1]中并且会逐步将ht[0]的数据转移到ht[1].全部rehash完毕后,ht[1]赋值给ht[0]然后清空ht[1].

模拟问题

0x00 迭代过程中,没有进行rehash

这个过程比较简单,一般来说只需要最简单粗暴的顺序迭代就可以了,这种情况下没什么好说的。

0x01 迭代过程中,进行过rehash

但是字典的大小是能够进行自动扩容的,我们不得不考虑以下两个问题:

第一,假如字典扩容了,变成2倍的长度,这种情况下,能够保证一定能遍历所有最初的key,但是却会出现大量重复。举个例子:

比如当前的key数组大小是4,后来变为8了。假如从下表0,1,2,3顺序扫描时,如果数组已经发生扩容,那么前面的0,1,2,3slot里面的数据会发生一部分迁移到对应的4,5,6,7slot里面去,当扫描到4,5,6,7的slot时,无疑会出现值重复的情况。

需要知道的是,Redis按如下方法计算一个当前key扩容后的slot:hash(key)&(size-1)

如图,当从字典大小从4扩容到8时,原先在0 slot的数据会分散到0(000)与4(100)两个slot,对应关系表如下:

第二, 如果字典缩小了,比如从16缩小到8, 原先scan已经遍历了0,1,2,3 ,如果数组已经缩小。这样后来迭代停止在7号slot,但是8,9,10,11这几个slot的数据会分别合并到0,1,2,3里面去,从而scan就没有扫描出这部分元素出来,无法保证可用性。

0x10 迭代过程中,正在进行rehash

上面考虑的情况是,在迭代过程的间隙中,rehash已经完成。那么会不会出现迭代进行中,切换游标时,rehash也正在进行?当然可能会发生。

如果返回游标1时正在进行rehash,那么ht[0](扩容之前的表)中的slot1中的部分数据可能已经rehash到 ht[1](扩容之后的表)中的slot1或者slot4,此时必须将ht[0]和ht[1]中的相应slot全部遍历,否则可能会有遗漏数据,但是这么做好像也非常麻烦。

解决方法

为了解决以上两个问题,Redis使用了一种称为:reverse binary iteration的算法。源码如下:

unsigned long dictScan(dict *d, unsigned long v, dictScanFunction *fn, void *privdata){ if (!dictIsRehashing(d)) { t0 = (d->ht[0]); m0 = t0->sizemask; /* Emit entries at cursor */ while (de) { fn(privdata, de); de = de->next; } } else { m0 = t0->sizemask; m1 = t1->sizemask; de = t0->table[v & m0]; while (de) { fn(privdata, de); de = de->next; } do { de = t1->table[v & m1]; while (de) { fn(privdata, de); de = de->next; } v = (((v | m0) + 1) & ~m0) | (v & m0); } while (v & (m0 ^ m1)); } v |= ~m0; v = rev(v); v++; v = rev(v); return v;}

一起来理解下核心源码,第一个if,else主要通过dictIsRehashing这个函数来判断是否正在rehash;sizemask指的是字典空间长度,假如长度为16,那么sizemask的二进制为00001111。m0 代表当前字典的长度,v代表游标所在的索引值。接下来关注这个片段:

v |= ~m0;v = rev(v);v++;v = rev(v);

这段代码初看好像有点摸不着头脑,怎么多次在多次rev?我们来看下在字典长度从4 rehash到8时,scan是如何迭代的。

当字典长度为4时,m0等于4,二进制表示为00000011,那么~m0为11111100,v初始值为0,那么 v |= ~m0为11111100。接下来看图:

可以看到,第一次dictScan后,游标从0变成了2,四次遍历分别为 0 -> 2 -> 1 -> 3,四个值都遍历到了。

在字典长度为8时,遍历情况如下:

遍历顺序为:0 -> 4 -> 2 -> 6 -> 1 -> 5 -> 3 -> 7

是不是察觉了什么?遍历顺序是不是顺序是一致的?如果还没发现,不妨再来看看字典长度为16时的遍历情况,以及三次顺序的对比:

让我们设想这么一个情况,字典的大小本身为4,开始迭代,当游标刚迭代完slot0时,返回的下一个游标时slot2,此时发现字典的大小已经从4rehash到8,那么不妨继续从size为8的hashtable中slot2处继续迭代。有人会说,不是把slot4遗漏掉了吗?

注意之前所说的扩容方式:hash(key)&(size-1),slot0和slot4的内容是相同的,巧妙地避开了重复,当然,更不会遗漏。

如果你看到这里,你可能会发出和我一样的感慨:我X,这算法太牛X了。没错,上面的算法是由Pieter Noordhuis设计实现的,Redis之父Salvatore Sanfilippo对该算法的评价是”Hard to explain but awesome。“

字典扩大的情况没问题,那么缩小的情况呢?可以仿照着自己思考一下具体步骤。答案是可能会出现重复迭代,但是不会出现遗漏,也能够保证可用性。

迭代过程中,进行过rehash这种情况下的迭代已经比较完美地解决了,那么迭代过程中,正在进行rehash的情况是如何解决的呢?

我们继续看源码,之前提到过dictIsRehashing这个函数用来判断是否正在进行rehash,那么主要就是关注这段源码:

 m0 = t0->sizemask; m1 = t1->sizemask; de = t0->table[v & m0]; while (de) { fn(privdata, de); de = de->next; } do { de = t1->table[v & m1]; while (de) { fn(privdata, de); de = de->next; } v = (((v | m0) + 1) & ~m0) | (v & m0); } while (v & (m0 ^ m1));

m0代表rehash前的字典长度,假设为4,即00000011,m1代表rehash后的字典长度,假设为8,即00000111。

首先当前游标 &m0可以得到较小字典中需要迭代的slot的索引,然后开始循环迭代。

然后开始较大字典的迭代,首先我们关注一下循环条件:

v & (m0 ^ m1)

m0,m1二者经过异或操作后的值为00000100,可以看到只留下了最高位的值。游标v与之做 &操作,将其作为判断条件,即判断游标v在最高位是否还有值。当高位为0时,说明较大字典已经迭代完毕。(因为较大字典的大小是较小字典的两倍,较大字典大小的最高位一定是1)

到此为止,我们已经将scan的核心源码通读一遍了,相信很多其间的迷惑也随之解开。不仅在写代码的时候更自信了,假如日后被面试官问起相关问题,那绝对可以趁机表现一番,想想还有点小激动。

转载:
https://mp.weixin.qq.com/s/s7143MZPz3HpZqr8TT23gA

相关推荐

为何说 :has() 选择器是对CSS架构的重塑?

大家好,很高兴又见面了,我是"...

不得不知的网络安全知识(网络 安全知识)

本文最初发布于BitsandPieces...

从凌晨发布的Manus到3小时复刻的OpenManus

前言2025/3/5凌晨一点半前后,手机里陆续收到一些公众号推文,Manus,LeaveittoManus,感觉一片喧哗,有点Agent里面的Deepseek类似赶脚,索性去注册了,Sorry...

JAVA安全加密通信协议详解(java密码加密哪种方式最安全)

JAVA安全加密通信协议详解在当今这个数字化时代,数据安全变得越来越重要。无论是在企业内部还是互联网上,保护敏感信息免受恶意攻击都是一项至关重要的任务。因此,了解并掌握安全加密通信协议对于每一位开发者...

两篇文章介绍:全新Swift从入门到进阶实战探探iOS APP(完结)

"夏哉ke":quangneng.com/5131/《全新Swift从入门到进阶实战探探iOSAPP》这一课程或书籍主要聚焦于使用Swift语言来开发iOS应用程序,尤其以“探探”这样的社交应用作为...

你写的JSP代码正在拖垮系统90%开发者不知道的过时陷阱与重生法则

"2024年了,我的团队还在用JSP!"某电商平台凌晨崩溃的监控警报,竟源自一行20年前的JSP代码逻辑。这个曾经统治JavaWeb的技术,正在用最隐蔽的方式摧毁你的系统性能......

Java jakarta常用注解详解(java validate注解)

持久化注解JakartaPersistence注解是...

网页出现 403 forbidden 是什么意思?

网页出现403forbidden是什么意思?“403forbidden”是一个HTTP状态码(HTTPSTATUSCODE),它的含义非常好理解。就是:...

孝琳《Queendom2》实力碾压难超越!韩网跪了:其他团来争第2的

记者刘宛欣/综合报导南韩女团竞赛节目《Queendom2》正如火如荼地进行中,日前播出6组人马的第二次竞演,其中以SOLO出击的孝琳继首次竞演拿下满分登冠军宝座后,第二次竞演又毫无悬念满分获得第一,...

《Queendom》排名公开(G)I—DLE夺第一 mamamoo展超强实力位列亚军

人民网讯5日,韩国最新综艺节目《Queendom》播出并公开了竞演排名。在本期节目中,人气女团OHMYGIRL演唱了《秘密花园》,(G)I-DLE将《LATATA》唱出十足底气,Lovelyz则...

React 4、路由库react-router-dom

在React应用中,路由(Routing)是一个关键的功能,它允许用户在不同页面或视图之间导航,而无需重新加载整个页面。React本身并不提供内置的路由功能,但你可以使用第三方库来实现路由。最常...

美人计 | 不管你追哪个团,都去看《Queendom》

以后也别选新的女团了,就让实力与美貌并存的姐姐们神仙打架就挺好。这个综艺8月底开播,最新一期是在十一假期的时候播出,它是Mnet推出的以女团同时回归对决的新概念综艺。参赛的6组都是已出道并且成绩相当不...

理念:无冲突的扩展本地DOM原型(俄乌冲突最新消息)

正如我昨天在博文中指出,我不喜欢使用jQuery的原因之一是因为它的包装对象。对于jQuery来说,这是一个明智的决定:早在2006年它被第一次开发出来的时候,IE有一个非常讨厌的内存泄漏bug,当我...

虚拟DOM真的比操作原生DOM快吗?前端大神提供4个参考观点!收藏

尤雨溪:https://www.zhihu.com/question/31809713/answer/53544875...

DOM丨睡不好伤肾更伤肝真的有证据!

点击蓝字...

取消回复欢迎 发表评论: