Redis数据结构及如何用好这些数据结构

这篇文章延续精简的风格,争取能让大家8min内读完并有所收获。

概述

一周前的文章精简的介绍了Redis核心原理及Redis是如何工作的,这一章回归到Redis的本质上来(存储)。Redis之所以如此受欢迎,一方面是性能强劲,另一方面就是支持多种常见的数据结构,甚至能完成一定的数据运算工作,灵活性大大提升。

Redis的常用四大数据结构类型,将一一介绍

1)String

2)List

3)Hash

4)Set


Redis数据结构


在文章Redis核心原理中介绍了,Redis字典的KV都是由redisObject组成的,每种Redis数据类型都是由不同的编码方式和真实的底层数据类型(ptr)实现,本文先介绍底层数据实现


SDS


Redis string底层实现

struct sdshdr {    

    // 记录 buf 数组中已使用字节的数量

    // 等于 SDS 所保存字符串的长度    

    int len;    

    // 记录 buf 数组中未使用字节的数量    

    int free;    

    // 字节数组,用于保存字符串    

    char buf[];

};

1)获取字符串长度的复杂度是O(1)

2)执行append操作时,如果空间不足则自动分配新空间,不会导致buf溢出

3)当字符串长度小于1MB时,将分配与len相同大小的未使用空间(如图所示),目的是在空间占用和空间重分配之间取得平衡


LINKEDLIST


Redis链表实现

1)list是双向链表,RPUSH LPUSH由此实现

2)获取链表长度为O(1)


HASH TABLE

Redis 哈希表实现

1)获取hash表size/used的复杂度为O(1)

2)有两张hash表组成,如果没有rehash状态下,rehashidx为-1,且其中的一张表是空的。当rehash时,为了不影响redis的性能,同时使用两张表渐进式的完成rehash:设置rehashidx为0,当查询或修改key时把KV转移到新表上去,直至旧表为空,设置rehashidx为-1,完成迁移

3)为什么hash table的长度是2的N次幂?因为bucket = hashcode & (len-1),当len为2的N次幂时len-1=1111。。  因此原有的对象只有少部分会被重新分配到新的bucket中去,因为(len-1) -> (2*len-1)时只有最高位的0变成了1,会减少迁移的工作量,提高效率

INT SET


Redis的整数集合

1)获取set length的复杂度为O(1)

2)encoding表明取值范围及内存占用,因此Redis并不适合将少量过大的数和大量小数值存在一起,会造成空间浪费

ZIPLIST


Redis的压缩表

1)ziplist是链表和哈希表的底层实现之一

2)ziplist采用更紧凑的编码,减少内存使用

以下是Redis的数据结构全貌

String

Redis字符串对象

List


ziplist实现的List


linked list实现的List

1)当列表中的object数量小于512个时,redis使用ziplist这种占用空间更小的结构

Hash


ziplist实现的哈希对象
ziplist实现的哈希对象


hashtable实现的哈希对象

1)当哈希表中的object数量小于512个时,Redis使用ziplist这种占用空间更小的结构。虽然复杂度为O(N),但由于object数量少,不会对CPU造成影响

Set


intset只包含数值类型的set


其它set

1)当一个set中全是数字时,采用intset为底层,占用空间小,因为尽量不要把少数string和大量int存储在一个set里

总结

1)使用哈希表、列表及集合时,尽量不要使用big data及添加过多的数据,因为这样会导致存储结构从ziplist变为hash table、linkedlist等空间占用更大的数据结构

2)如无必要的情况下,一个集合中不要将string和int混合存储,这会导致内存占用变大

3)String不要过长,小于39字符时,使用embstr编码格式,占用更少内存

4)set可以用来做数据运算,如交、并等大量数据的运算

参考文档

Redis设计与实现

Redis原理及应用

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,029评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,238评论 3 388
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,576评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,214评论 1 287
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,324评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,392评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,416评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,196评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,631评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,919评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,090评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,767评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,410评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,090评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,328评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,952评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,979评论 2 351

推荐阅读更多精彩内容