渐进式 rehash

上一节说过, 扩展或收缩哈希表需要将 ht[0] 里面的所有键值对 rehash 到 ht[1] 里面, 但是, 这个 rehash 动作并不是一次性、集中式地完成的, 而是分多次、渐进式地完成的。

这样做的原因在于, 如果 ht[0] 里只保存着四个键值对, 那么服务器可以在瞬间就将这些键值对全部 rehash 到 ht[1] ; 但是, 如果哈希表里保存的键值对数量不是四个, 而是四百万、四千万甚至四亿个键值对, 那么要一次性将这些键值对全部 rehash 到 ht[1] 的话, 庞大的计算量可能会导致服务器在一段时间内停止服务。

因此, 为了避免 rehash 对服务器性能造成影响, 服务器不是一次性将 ht[0] 里面的所有键值对全部 rehash 到 ht[1] , 而是分多次、渐进式地将 ht[0] 里面的键值对慢慢地 rehash 到 ht[1] 。

以下是哈希表渐进式 rehash 的详细步骤:

  1. 为 ht[1] 分配空间, 让字典同时持有 ht[0] 和 ht[1] 两个哈希表。
  2. 在字典中维持一个索引计数器变量 rehashidx , 并将它的值设置为 0 , 表示 rehash 工作正式开始。
  3. 在 rehash 进行期间, 每次对字典执行添加、删除、查找或者更新操作时, 程序除了执行指定的操作以外, 还会顺带将 ht[0] 哈希表在 rehashidx 索引上的所有键值对 rehash 到 ht[1] , 当 rehash 工作完成之后, 程序将 rehashidx 属性的值增一。
  4. 随着字典操作的不断执行, 最终在某个时间点上, ht[0] 的所有键值对都会被 rehash 至 ht[1] , 这时程序将 rehashidx 属性的值设为 -1 , 表示 rehash 操作已完成。

渐进式 rehash 的好处在于它采取分而治之的方式, 将 rehash 键值对所需的计算工作均滩到对字典的每个添加、删除、查找和更新操作上, 从而避免了集中式 rehash 而带来的庞大计算量。

图 4-12 至图 4-17 展示了一次完整的渐进式 rehash 过程, 注意观察在整个 rehash 过程中, 字典的 rehashidx 属性是如何变化的。

渐进式 rehash 执行期间的哈希表操作

因为在进行渐进式 rehash 的过程中, 字典会同时使用 ht[0] 和 ht[1] 两个哈希表, 所以在渐进式 rehash 进行期间, 字典的删除(delete)、查找(find)、更新(update)等操作会在两个哈希表上进行: 比如说, 要在字典里面查找一个键的话, 程序会先在 ht[0] 里面进行查找, 如果没找到的话, 就会继续到 ht[1] 里面进行查找, 诸如此类。

另外, 在渐进式 rehash 执行期间, 新添加到字典的键值对一律会被保存到 ht[1] 里面, 而 ht[0] 则不再进行任何添加操作: 这一措施保证了 ht[0] 包含的键值对数量会只减不增, 并随着 rehash 操作的执行而最终变成空表。

总结

  • 字典被广泛用于实现 Redis 的各种功能, 其中包括数据库和哈希键。
  • Redis 中的字典使用哈希表作为底层实现, 每个字典带有两个哈希表, 一个用于平时使用, 另一个仅在进行 rehash 时使用。
  • 当字典被用作数据库的底层实现, 或者哈希键的底层实现时, Redis 使用 MurmurHash2 算法来计算键的哈希值。
  • 哈希表使用链地址法来解决键冲突, 被分配到同一个索引上的多个键值对会连接成一个单向链表。
  • 在对哈希表进行扩展或者收缩操作时, 程序需要将现有哈希表包含的所有键值对 rehash 到新哈希表里面, 并且这个 rehash 过程并不是一次性地完成的, 而是渐进式地完成的。

Redis之渐进式rehash相关推荐

  1. Redis的渐进式Rehash

    为了实现从键到值的快速访问,Redis 使用了一个哈希表来保存所有键值对. 一个哈希表,其实是一个数组,数组的每个元素称为一个哈希桶. 哈希桶中的 entry 元素中保存了 *key 和 *value ...

  2. 【Redis】什么是渐进式rehash

    字典 字典是一种用于保存键值对的抽象数据结构.比如set msg "hello redis"这个键值对就是保存在数据库字典中 字典还是哈希键的底层实现之一,当一个哈希键包含的键值对 ...

  3. Redis:缩容、扩容、渐进式rehash

    目录 1.缩容 扩容 2.渐进式rehash 1.缩容 扩容 随着redis的操作的不断执行,哈希表保存的键值会逐渐地增多或者减少,为了让哈希表的负载因子(ratio)维持在一个合理的范围之内,当哈希 ...

  4. redis渐进式rehash机制

    在Redis中,键值对(Key-Value Pair)存储方式是由字典(Dict)保存的,而字典底层是通过哈希表来实现的.通过哈希表中的节点保存字典中的键值对.我们知道当HashMap中由于Hash冲 ...

  5. 单线程照样飞起 | redis字典快速映射+hash釜底抽薪+渐进式rehash

    前言 相信你一定使用过新华字典吧!小时候不会读的字都是通过字典去查找的.在Redis中也存在相同功能叫做字典又称为符号表!是一种保存键值对的抽象数据结构 本篇仍然定位在[redis前传]系列中,因为本 ...

  6. Redis详解(六)渐进式rehash机制

    在Redis中,键值对(Key-Value Pair)存储方式是由字典(Dict)保存的,而字典底层是通过哈希表来实现的.通过哈希表中的节点保存字典中的键值对.我们知道当HashMap中由于Hash冲 ...

  7. redis中的hash扩容渐进式rehash过程

    背景: redis字典(hash表)当数据越来越多的时候,就会发生扩容,也就是rehash 对比:java中的hashmap,当数据数量达到阈值的时候(0.75),就会发生rehash,hash表长度 ...

  8. hash地址_redis中的hash扩容、渐进式rehash过程

    背景: redis字典(hash表)当数据越来越多的时候,就会发生扩容,也就是rehash 对比:java中的hashmap,当数据数量达到阈值的时候(0.75),就会发生rehash,hash表长度 ...

  9. 四、redis的底层数据结构

    1.演示数据类型的实现 上篇博客我们在介绍 key 相关命令的时候,介绍了如下命令: OBJECT ENCODING key 该命令是用来显示那五大数据类型的底层数据结构. 比如对于 string 数 ...

最新文章

  1. Android性能优化常见问题,终局之战
  2. Seq2Seq之双向解码机制 | 附开源实现
  3. 用Java描述数据结构之线性表的顺序存储(顺序表),ArrayList及其方法的介绍
  4. sysadmin默认密码_从sysadmin过渡到DevOps工程师的案例
  5. 小程序获取用户地址信息api
  6. 基于40万表格数据集TableBank,用MaskRCNN做表格检测
  7. 韩开发新技术 用纸代替硅制造电路芯片
  8. SM3算法的C++实现(代码)
  9. 计算机自带游戏如何删除,win7怎么彻底删除自带游戏,win7自带游戏怎么删除隐藏...
  10. 12306系统升级对电力营销系统改造的启示
  11. 推荐 10 款让你的 Chrome 浏览器功能更强大的插件
  12. oracle10g 管理共享服务器
  13. java 配置文件加密_Spring cloud config 配置文件加密方式
  14. Atitit 理财之道---分期与利率的比较列表 attilax总结
  15. 股市低迷 期权策略表现突出
  16. php怎么判断是否手机网站,php,js判断网站是不是手机访问
  17. 如何使用GridView实现不同数量显示图片不同的大小
  18. LearnOpenGL_study -- 光照贴图
  19. 学术论文(paper)投稿格式规范
  20. 当当网 用什么数据库 mysql_当当网爬取数据到MySQL数据库,没有信息。

热门文章

  1. jquery关于checkbox在ie下的一个bug
  2. 微信公众号配置后台接入
  3. 关于LabVIEW视觉ROI的读取与存储
  4. 【题解】牛客小白月赛16(部分题,待补充……)
  5. 高考成绩查询2021年莆田市,2021年莆田高考志愿填报系统入口
  6. Shell命令-文件及内容处理之grep(egrep)、join
  7. postgres预写式日志的内核实现详解-wal记录读取
  8. 性能测试过程中oracle数据库报ORA-27301 ORA-27302错
  9. Python3标准库built-in、itertools、functools中的生成器
  10. flask简单登录注册