Redis源码学习之【哈希字典】

介绍

Redis的哈希字典通过key值来找对应的value。需要注意的是Redis的字典是如何进行rehash的。

源码

dict.h dict.c

数据结构


如上图所示,哈希字典用dict结构体表示,其中含有两个哈希表,主要用于进行rehash操作。同时哈希表使用量表的方式解决冲突。具体的数据结构如下:

  1. /* 
  2.  * 哈希表节点 
  3.  */  
  4. typedef struct dictEntry {  
  5.   
  6.     // 键  
  7.     void *key;  
  8.   
  9.     // 值  
  10.     union {  
  11.         void *val;  
  12.         uint64_t u64;  
  13.         int64_t s64;  
  14.     } v;  
  15.   
  16.     // 链往后继节点  
  17.     struct dictEntry *next;   
  18.   
  19. } dictEntry;  
  20.   
  21. /* 
  22.  * 特定于类型的一簇处理函数 
  23.  */  
  24. typedef struct dictType {  
  25.     // 计算键的哈希值函数  
  26.     unsigned int (*hashFunction)(const void *key);  
  27.     // 复制键的函数  
  28.     void *(*keyDup)(void *privdata, const void *key);  
  29.     // 复制值的函数  
  30.     void *(*valDup)(void *privdata, const void *obj);  
  31.     // 对比两个键的函数  
  32.     int (*keyCompare)(void *privdata, const void *key1, const void *key2);  
  33.     // 键的释构函数  
  34.     void (*keyDestructor)(void *privdata, void *key);  
  35.     // 值的释构函数  
  36.     void (*valDestructor)(void *privdata, void *obj);  
  37. } dictType;  
  38.   
  39. /* 
  40.  * 哈希表 
  41.  */  
  42. typedef struct dictht {  
  43.   
  44.     // 哈希表节点指针数组(俗称桶,bucket)  
  45.     dictEntry **table;        
  46.   
  47.     // 指针数组的大小  
  48.     unsigned long size;       
  49.   
  50.     // 指针数组的长度掩码,用于计算索引值  
  51.     unsigned long sizemask;   
  52.   
  53.     // 哈希表现有的节点数量  
  54.     unsigned long used;       
  55.   
  56. } dictht;  
  57.   
  58. /* 
  59.  * 字典 
  60.  * 
  61.  * 每个字典使用两个哈希表,用于实现渐进式 rehash 
  62.  */  
  63. typedef struct dict {  
  64.   
  65.     // 特定于类型的处理函数  
  66.     dictType *type;  
  67.   
  68.     // 类型处理函数的私有数据  
  69.     void *privdata;  
  70.   
  71.     // 哈希表(2个)  
  72.     dictht ht[2];         
  73.   
  74.     // 记录 rehash 进度的标志,值为-1 表示 rehash 未进行  
  75.     int rehashidx;  
  76.   
  77.     // 当前正在运作的安全迭代器数量  
  78.     int iterators;        
  79.   
  80. } dict;  
  81.   
  82. /* 
  83.  * 字典迭代器 
  84.  * 
  85.  * 如果 safe 属性的值为 1 ,那么表示这个迭代器是一个安全迭代器。 
  86.  * 当安全迭代器正在迭代一个字典时,该字典仍然可以调用 dictAdd 、 dictFind 和其他函数。 
  87.  * 
  88.  * 如果 safe 属性的值为 0 ,那么表示这不是一个安全迭代器。 
  89.  * 如果正在运作的迭代器是不安全迭代器,那么它只可以对字典调用 dictNext 函数。 
  90.  */  
  91. typedef struct dictIterator {  
  92.   
  93.     // 正在迭代的字典  
  94.     dict *d;                  
  95.   
  96.     int table,              // 正在迭代的哈希表的号码(0 或者 1)  
  97.         index,              // 正在迭代的哈希表数组的索引  
  98.         safe;               // 是否安全?  
  99.   
  100.     dictEntry *entry,       // 当前哈希节点  
  101.               *nextEntry;   // 当前哈希节点的后继节点  
  102. } dictIterator;  

分析

rehash

在字典中使用了两个hash表就是为了方便进行rehash的,rehash主要有两种方式,一是由后台调用cron进行按照100步进进行hash,二是在进行添加删除字典中的元素的时候进行1步的hash。

这里每一次的hash的步进是以真个hash key对应的链表为单位的,也就是说1步进的rehash是将原来hash表中的一个key链表进行rehash到新的哈希表中的位置。这里可以看出Redis在此处是牺牲了内存但是换来了性能的响应。将rehash进行分散操作可以避免阻塞导致的性能急剧下降。

迭代器

迭代器的属性如果是安全的则表明可以在迭代的过程中进行dictAdd等其他的操作,而如果迭代器是不安全的则只能进行next的操作。

同时Redis限制在有迭代器访问字典的时候进行rehash,因为这样会造成对一个元素访问多次。但是在rehash的过程中可以随时的对字典进行遍历,一旦迭代器开始遍历字典了,rehash就会暂停知道没有迭代器在对字典进行遍历为止。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
资源包主要包含以下内容: ASP项目源码:每个资源包中都包含完整的ASP项目源码,这些源码采用了经典的ASP技术开发,结构清晰、注释详细,帮助用户轻松理解整个项目的逻辑和实现方式。通过这些源码,用户可以学习到ASP的基本语法、服务器端脚本编写方法、数据库操作、用户权限管理等关键技术。 数据库设计文件:为了方便用户更好地理解系统的后台逻辑,每个项目中都附带了完整的数据库设计文件。这些文件通常包括数据库结构图、数据表设计文档,以及示例数据SQL脚本。用户可以通过这些文件快速搭建项目所需的数据库环境,并了解各个数据表之间的关系和作用。 详细的开发文档:每个资源包都附有详细的开发文档,文档内容包括项目背景介绍、功能模块说明、系统流程图、用户界面设计以及关键代码解析等。这些文档为用户提供了深入的学习材料,使得即便是从零开始的开发者也能逐步掌握项目开发的全过程。 项目演示与使用指南:为帮助用户更好地理解和使用这些ASP项目,每个资源包中都包含项目的演示文件和使用指南。演示文件通常以视频或图文形式展示项目的主要功能和操作流程,使用指南则详细说明了如何配置开发环境、部署项目以及常见问题的解决方法。 毕业设计参考:对于正在准备毕业设计的学生来说,这些资源包是绝佳的参考材料。每个项目不仅功能完善、结构清晰,还符合常见的毕业设计要求和标准。通过这些项目,学生可以学习到如何从零开始构建一个完整的Web系统,并积累丰富的项目经验。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值