深入学习Redisdict源码剖析(redis源码之dict)

深入学习Redis:dict源码剖析

Redis是一个高性能的键值存储系统,其中的数据结构dict扮演了至关重要的角色。dict是Redis内部用来实现哈希表的数据结构,它的性能决定了Redis的性能。本文将深入探究Redis中的dict数据结构源码以及如何使用它。

dict的基本结构

在Redis的源码中,dict的实现在字典结构体dictType中定义。dictType的结构如下:

typedef struct dictType {
unsigned int (*hashFunction)(const void *key);
void *(*keyDup)(void *privdata, const void *key);
void *(*valDup)(void *privdata, const void *obj);
int (*keyCompare)(void *privdata, const void *key1, const void *key2);
void (*keyDestructor)(void *privdata, void *key);
void (*valDestructor)(void *privdata, void *obj);
} dictType;

dictType中定义了dict的6个回调函数,包括了键的哈希函数、键的复制函数、值的复制函数、键的比较函数、键的析构函数以及值的析构函数。

接着我们看dictEntry的定义,dictEntry正好能代表dict中的一项,其结构体如下:

typedef struct dictEntry {
void *key;
union {
void *val;
uint64_t u64;
int64_t s64;
double d;
} v;
struct dictEntry *next;
} dictEntry;

dictEntry中存放了一个key以及一个val,其中val是一个Union,能存放多种类型的值。next是为了解决hash算法冲突而存在的指针。

最后是dict的定义了,它的结构体定义如下:

typedef struct dictht {
dictEntry **table;
unsigned long size;
unsigned long sizemask;
unsigned long used;
} dictht;

typedef struct dict {
dictType *type;
void *privdata;
dictht ht[2];
long rehashidx; /* rehashing not in progress if rehashidx == -1 */
int iterators; /* number of iterators currently running */
} dict;

在dict的定义中,我们可以看到dict是由两个dictht结构体组成的,也就是说,dict中包含了两个哈希表(table)。其中一个叫做ht[0],它是正在被使用的哈希表,ht[1]则是用来在rehash(rehash是Redis用来动态扩容的)过程中存放数据的哈希表。

dict的方法实现

dict的方法可以分为两类:通用方法和私有方法。通用方法是指dict的外部接口,提供给外部程序使用的方法。它们包括dictAdd、dictFind、dictDelete等方法。而私有方法是指dict内部使用的方法,只供dict内部调用。它们包括_dictRehashStep、_dictKeyIndex、_dictExpandIfNeeded等方法。

dict的添加函数dictAdd的实现如下:

int dictAdd(dict *ht, void *key, void *val)
{
dictEntry *entry = dictAddRaw(ht,key,NULL);
if (!entry) return DICT_ERR;
dictSetVal(ht, entry, val);
return DICT_OK;
}

dictAdd方法调用了dictAddRaw方法来添加一个新的键值对,如果添加成功,则返回DICT_OK,否则返回DICT_ERR。

dict的查找函数dictFind实现如下:

dictEntry *dictFind(dict *ht, const void *key)
{
dictEntry *he;
unsigned int h;

if (ht->size == 0) return NULL;
if (dictIsRehashing(ht)) _dictRehashStep(ht);
h = dictHashKey(ht, key);
for (int table = 0; table
he = ht->ht[table].table[h & ht->ht[table].sizemask];
while (he) {
if (dictCompareKeys(ht, key, he->key))
return he;
he = he->next;
}
if (!dictIsRehashing(ht)) return NULL;
}
return NULL;
}

dictFind方法通过dictHashKey方法计算出key对应的哈希值,然后从两个哈希表中查找对应的dictEntry对象。查找时,先从正在使用的哈希表(table[0])开始查找,如果找到了就返回当前的dictEntry对象,如果在table[0]中没有找到,再到table[1]中查找。如果在table[1]中依然没有找到,说明该key不存在于Redis里。

dict的删除函数dictDelete实现如下:

int dictDelete(dict *ht, const void *key)
{
dictEntry *he, *prevHe = NULL;
unsigned int h;
int table;
if (ht->ht[0].size == 0 && ht->ht[1].size == 0) return DICT_ERR;
if (dictIsRehashing(ht)) _dictRehashStep(ht);
h = dictHashKey(ht, key);

for (table = 0; table
he = ht->ht[table].table[h & ht->ht[table].sizemask];
while (he) {
if (dictCompareKeys(ht, key, he->key)) {
if (prevHe)
prevHe->next = he->next;
else
ht->ht[table].table[h & ht->ht[table].sizemask] = he->next;
dictFreeEntryKey(ht, he);
dictFreeEntryVal(ht, he);
zfree(he);
ht->ht[table].used--;
return DICT_OK;
}
prevHe = he;
he = he->next;
}
if (!dictIsRehashing(ht)) break;
}
return DICT_ERR;
}

dictDelete方法先计算key的哈希值,然后从两个哈希表中查找包含该key的dictEntry对象。找到该对象后,从链表上删除,并释放内存。如果删除成功,返回DICT_OK,否则返回DICT_ERR。

使用dict实现缓存机制

dict的高效性能为Redis实现缓存机制提供了极大的便利。我们可以将常用数据放入Redis中,它们会被存储在内存中,当需要访问数据时可以直接从内存中读取,减少了文件I/O操作,提升了访问速度。

我们可以编写以下代码来实现缓存机制:

#include "redis.h"
void cacheData(redisClient *c) {
dictEntry *de;
dict *dictPtr;
long long bytes = 0;
/* If the key already exists in the cache, delete it first */
if (lookupKeyRead(c->db,c->argv[1]->ptr) != NULL) {
dbDelete(c->db,c->argv[1]);
}
/* Add the key-value pr to the cache */
dictPtr = c->db->dict;
de = dictAddRaw(dictPtr,c->argv[1]->ptr,NULL);
bytes += sdslen(c->argv[1]->ptr)+dictSize(dictPtr->valsize);
dictSetVal(dictPtr, de, c->argv[2]);
incrRefCount(c->argv[2]);
c->db->dict->expire = 0;
addReply(c,shared.ok);
}

cacheData方法首先从Redis中查找key,如果已经存在这个key,则需要先将这个key对应的value从Redis中删除。然后,将新的key-value放入Redis的dict中即可。

本文简要的介绍了dict的基本结构以及dict的三个基本方法,同时也讲述了如何使用dict实现缓存机制。在实际的应用中,我们还需要根据实际情况合理的设置dict的回调函数,以让dict提供更好的性能,从而为Redis提供高效的数据存储和检索服务。

香港服务器首选树叶云,2H2G首月10元开通。
树叶云(www.IDC.Net)提供简单好用,价格厚道的香港/美国云服务器和独立服务器。IDC+ISP+ICP资质。ARIN和APNIC会员。成熟技术团队15年行业经验。

文章来源网络,作者:运维,如若转载,请注明出处:https://shuyeidc.com/wp/297371.html<

(0)
运维的头像运维
上一篇2025-05-22 10:50
下一篇 2025-05-22 10:51

相关推荐

  • 个人主题怎么制作?

    制作个人主题是一个将个人风格、兴趣或专业领域转化为视觉化或结构化内容的过程,无论是用于个人博客、作品集、社交媒体账号还是品牌形象,核心都是围绕“个人特色”展开,以下从定位、内容规划、视觉设计、技术实现四个维度,详细拆解制作个人主题的完整流程,明确主题定位:找到个人特色的核心主题定位是所有工作的起点,需要先回答……

    2025-11-20
    0
  • 社群营销管理关键是什么?

    社群营销的核心在于通过建立有温度、有价值、有归属感的社群,实现用户留存、转化和品牌传播,其管理需贯穿“目标定位-内容运营-用户互动-数据驱动-风险控制”全流程,以下从五个维度展开详细说明:明确社群定位与目标社群管理的首要任务是精准定位,需明确社群的核心价值(如行业交流、产品使用指导、兴趣分享等)、目标用户画像……

    2025-11-20
    0
  • 香港公司网站备案需要什么材料?

    香港公司进行网站备案是一个涉及多部门协调、流程相对严谨的过程,尤其需兼顾中国内地与香港两地的监管要求,由于香港公司注册地与中国内地不同,其网站若主要服务内地用户或使用内地服务器,需根据服务器位置、网站内容性质等,选择对应的备案路径(如工信部ICP备案或公安备案),以下从备案主体资格、流程步骤、材料准备、注意事项……

    2025-11-20
    0
  • 如何企业上云推广

    企业上云已成为数字化转型的核心战略,但推广过程中需结合行业特性、企业痛点与市场需求,构建系统性、多维度的推广体系,以下从市场定位、策略设计、执行落地及效果优化四个维度,详细拆解企业上云推广的实践路径,精准定位:明确目标企业与核心价值企业上云并非“一刀切”的方案,需先锁定目标客户群体,提炼差异化价值主张,客户分层……

    2025-11-20
    0
  • PS设计搜索框的实用技巧有哪些?

    在PS中设计一个美观且功能性的搜索框需要结合创意构思、视觉设计和用户体验考量,以下从设计思路、制作步骤、细节优化及交互预览等方面详细说明,帮助打造符合需求的搜索框,设计前的规划明确使用场景:根据网站或APP的整体风格确定搜索框的调性,例如极简风适合细线条和纯色,科技感适合渐变和发光效果,电商类则可能需要突出搜索……

    2025-11-20
    0

发表回复

您的邮箱地址不会被公开。必填项已用 * 标注