Redis整理

Redis简介

Redis是一个速度非常快的非关系数据库(non-relational database),它可以存储键(key)与5种不同类型的值(value)之间的映射(mapping),可以将存储在内存的键值对数据持久化到硬盘,可以使用复制特性来扩展读性能,还可以使用客户端分片来扩展写性能。

Redis与其他数据库和缓存服务器的对比

Redis与其他数据库和缓存服务器的对比

Redis 数据结构

SDS

SDS(simple dynamic string)简单动态字符串
SDS数据结构

  SDS的API都是二进制安全的(binary-safe),所有SDS API都会以处理二进制的方式来处理SDS存放在buf数组里的数据,程序不会对其中的数据做任何限制,过滤,或者假设数据在写入时是什么样的,它被读取时就是什么样。

链表

链表结构图

字典

字典hash表数据结构
字典数据结构
  • 使用Murmurhash2算法来计算键的哈希值
  • 使用链地址法来解决键冲突
  • rehash:随着操作的不断执行,哈希表保存的键值对会逐渐地增多或者减少,为了让hash表的负载因子(load factor)维持在一个合理的范围内,当hash表中保存的键值对数量太多或者太少时,程序需要对hash表的大小进行相应的扩展或者收缩,扩展或者收缩hash表的工作就是通过rehash(重新散列,hash函数还是原来的函数)操作来完成的;负载因子 = hash表已经保存节点数量 / hash表大小
  • 渐进式rehash:rehash动作不是一次性,集中式地完成的,而是分多次,渐进式地完成的。

跳跃表skiplist

  跳跃表skiplist是一种有序数据结构,它通过在每个节点中维持多个指向其他节点的指针,从而达到快速访问节点的目的,如下图所示:
跳跃表skiplist例子
  • 层(level): 节点中用L1,L2,L3等字样标记节点的各个层,L1代表第一层,L2代表第二层,一次类推。每个层都带有两个属性:前进指针和跨度,前进指针用于访问位于表尾方向的其他节点;而跨度则记录了前进指针所指向节点和当前节点的距离。
  • 分值:各个节点中的1.0,2.0,3.0是节点所保存的分值,在跳跃表中,节点按照各自所保存的分值从小到大排列。

整数集合

  整数集合(intset)是集合键的底层实现之一,当一个集合只包含整数值元素,并且这个集合的元素数量不多时,Redis就会使用整数集合作为集合键的底层实现,结构定义如下图所示:
数据结构定义
  • 有序性和唯一性:整数集合的每个元素都是contents数组的一个数组项(item),每个项在数组中按值的大小从小到大有序地排列,并且数组中不包含任何重复键
  • encoding属性决定了contents数组真正的类型,数组中的几个元素算作一个元素的问题,如下图所示:
    intset encoding例子说明
  • 升级:每当要将一个新元素添加到整数集合里面,并且新元素的类型比整数集合现有所有元素的类型都要长时,整数集合需要先进行升级(upgrade),然后才能将新元素添加到整数集合里面。

压缩列表ziplist

  压缩列表(ziplist)是列表键和哈希键的底层实现之一。当一个列表键只包含少量列表项,并且每个列表项要么就是小整数值,要么就是长度比较短的字符串,那么Redis就会使用压缩列表来做列表键的底层实现。压缩列表是Redis为了节约内存而开发的,是由一系列特殊编码的连续内存块组成的顺序型(sequential)数据结构,一个压缩列表可以包含任意多个节点(entry),每个节点可以保存一个字节数组或者一个整数值,如下图所示:
ziplist结构
  • 有连锁更新的问题:在特殊情况下产生的连续多次空间扩展操作称之为“连锁更新(cascade update)”。

对象

  Redis并没有直接使用这些数据结构来实现键值对数据库,而是基于这些数据结构创建了一个对象系统,包含如下的5中对象类型:
Redis对应键值的5中数据结构
  • 通过这五种不同类型的对象,Redis可以在执行命令之前,根据对象的类型来判断一个对象是否可以执行给定的命令
  • 内存回收机制:Redis的对象系统基于引用计数技术实现了内存回收机制;
  • 对象共享机制:Redis使用引用计数技术实现了对象共享机制。
       Redis使用对象来表示数据库中的键和值,每当我们在Redis数据库中新创建一个键值对时,至少会创建两个对象,一个对象用作键值对的键(键对象,键总是一个字符串对象),另一个对象用作键值对的值(值对象)。
数据结构 对象类型
数据结构
对象类型

字符串对象

  字符串对象的编码可以是int,raw或者embstr,这三种字符串编码在一定条件下会进行转换。

  • 如果一个字符串对象保存的是整数值,并且这个整数值可以用long类型来表示,那么字符串对象会将整数值保存在字符串对象结构的ptr属性中(将void转换为long),并将字符串对象的编码设置为int。

  • 如果字符串对象保存的是一个字符串的值,并且这个字符串值的长度大于32字节,那么字符串对象将使用一个简单动态字符串SDS来保存这个字符串的值,并将对象的编码设置为raw,如下图所示
    raw编码字符串对象结构
  • 如果字符串对象保存的是一个字符串值,并且这个字符串值的长度小于等于32字节,那么字符串对象将使用embstr(embstr编码是专门用于保存端字符串的一种优化编码方式)编码的方式来保存这个字符串值;这种编码和raw编码一样,都是用redisObject结构和sdshdr结构来表示字符串对象,但raw编码会调用两次内存分配函数来分别创建redisObject结构和sdshdr结构,而embstr编码则通过调用一次内存分配函数来分配一块连续的空间,空间中一次包含redisObject和sdshdr两个结构,如下图所示:

    embstr编码字符串结构

  • 字符串对象对应的命令实现:
    字符串对象对应的命令实现

列表对象

列表对象的编码可以是ziplist或者linkedlist;ziplist编码使用的压缩列表作为底层实现,linkedlist使用的是双端链表作为底层实现,这两种编码会在一定的条件自动进行转换,两种列表编码如下表对比所示:

ziplist 编码 linkedlist 编码
ziplist编码的列表对象
linkedlist编码的列表对象

列表命令实现入下图所示:
列表命令实现

哈希对象

哈希对象的编码可以ziplist或者hashtable;hashtable编码的哈希对象使用字典作为底层实现;这两种编码可以在一定的条件下相互转换;下表显示了两种编码的对比:

ziplist 编码 hashtable 编码
ziplist编码的哈希对象
hashtable编码的哈希对象

哈希对象的命令实现如下图所示:
哈希对象的命令实现

集合对象

  集合对象的编码可以是intset或者hashtable,intset编码的集合对象使用整数集合作为底层实现;这两种编码在一定条件下可以相互转换,对比如下图所示:

intset 编码 hashtable 编码
intset编码的集合对象
hashtable编码的集合对象

集合对象对应命令实现如下图所示:
集合对象的命令实现
集合对象的命令实现-续

有序集合

  有序集合的编码可以是ziplist或者skiplist(跳跃表),skiplist编码的有序集合对象使用zset结构作为底层实现,一个zset结构同时包含一个字典和一个跳跃表,如下代码所示:

typedef struct zset {
        zskiplist *zsl;
        dict *dict;
}

这两种编码在一定条件下可以相互转换,对比如下表所示:

ziplist 编码 skiplist 编码
ziplist编码的有序集合对象
skiplist编码的有序集合对象

有序集合对应的命令实现如下图所示:
有序集合命令实现

redis常用命令

TYPE命令

  对于Redis数据库保存的键值对来说,键总是一个字符串对象,而值则可以是字符串对象,列表对象,哈希对象,集合对象或者有序集合对象中一种,因此在对一个数据库键执行TYPE命令时,命令返回的结果为数据库键对应的值对象的类型,而不是键对象的类型,其中格式:type 键名称 ,对应的输出如下图所示:

不同类型值对象的TYPE命令输出

OBJECT ENCODING命令

对象的编码就是对象使用了什么数据结构作为对象的底层实现,格式为:OBJECT ENCODING 键名称,对应的输出下图所示:

OBJECT ENCODING对不同编码的输出
OBJECT ENCODING对不同编码的输出-续

OBJECT IDLETIME命令

   OBJECT IDLETIME命令可以打印出给定键的空转时间,空转时间=当前时间 - 键的值对象最后一次被命令程序访问的时间

对象通用命令

  • DEL命令
  • EXPIRE命令
  • EXPIREAT 命令
  • RENAME命令
  • SORT命令
  • TYPE命令
  • OBJECT命令
  • TTL命令:TTL命令以秒为单位返回键的剩余生存时间
  • PTTL命令:PTTL命令则以毫秒为单位返回键的剩余生存时间
  • SAVE命令:SAVE命令会阻塞Redis服务器进程,直到RDB文件创建完毕为止,在服务器进程阻塞期间,服务器不能处理任何命令请求
  • BGSAVE命令:BGSAVE命令会派生出一个子进程,然后由子进程负责创建RDB文件,服务器进程(父进程)继续处理命令请求
  • FLUSHALL命令:清除当前数据库中的所有键值对
  • CLIENT list命令:列出目前所有连接到服务器的普通客户端,命令输出中的fd域显示了服务器连接客户端所使用的套接字描述符
  • AUTH命令:如果authenticated的值为0,那么表示客户端未通过身份验证;如果authenticated的值为1,那么表示客户端已经通过了身份验证,AUTH命令可以是客户端的状态authenticated的值从0到1
  • PING命令
  • SLAVEOF命令:通过向从服务器发送SLAVEOF命令,可以让一个从服务器去复制一个主服务器,命令SLAVEOF <master_ip> <master_port>
  • EXISTS命令
  • INFO replication: 客户端向主服务器发送INFO replication命令得到的副本信息回复
  • redis-sentinel命令:redis-sentinel /path/to/your/sentinel.conf 或者 redis-server /path/to/your/sentinel.conf --sentinel
  • SUBSCRIBE命令:订阅
  • PSUBSCRIBE命令:订阅一个或多个模式
  • PUBLISH命令
  • 事务命令:MULTI/EXEC
  • WATCH命令:WATCH命令是一个乐观锁,它可以在EXEC命令执行之前,监视任意数量的数据库键,并在EXEC命令执行时,检查被监视的键是否至少有一个已经被修改过了,如果是的话,服务器将拒绝执行事务,并向客户端返回代表事务执行失败的空回复。
  • EVAL命令:EVAL命令可以直接对输入的脚本进行求值
  • EVALSHA命令:根据脚本的SHA1校验和来对脚本进行求值
  • SCRIPT LOAD命令:加载脚本
  • SETBITS/GETBITS/BITCOUNT/BITOP命令:位数组
  • CONFIG SET命令
  • SLOWLOG GET命令:查看服务器所保存的慢查询日志
  • MONITOR命令

select命令

  每个Redis客户端都有自己的目标数据库,每当客户端执行数据库写命令或者数据库读命令的时候,目标数据库就会成为这些命令的操作对象;默认情况下,Redis客户端的目标数据库是0号数据库,但客户端可以通过执行select命令来切换目标数据库;数据库对应的结构如下:

redisServer redisDb 示例图
redisServer
redisDb
示例图

数据库持久化

RDB持久化

RDB持久化功能是将Redis在内存中的数据库状态保存到磁盘里面,避免数据意外丢失,如下图所示:
RDB持久化

AOF持久化

AOF(append only File)持久化功能,与RDB持久化通过保存数据库中的键值对来记录数据库状态不同,AOF持久化是通过保存Redis服务器所执行的写命令来记录数据库状态的,如下图所示:
AOF持久化

参考

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

推荐阅读更多精彩内容

  • 今天是一个多年老友的生日,时光真的是让人猝不及防,一晃我们相识也有七八年,说到他也用多年老友来形容了。那些年他在我...
    薛定谔的猫没有鱼阅读 276评论 0 0
  • 感赏今天又有外财进账!我一直都运气那么好,领导对我也很偏爱,本来是正常的工作,只是急要一些,说是我加班辛苦了,非要...
    幸福的燕子阅读 127评论 0 0
  • 《熊先生的矛盾心情》 讨厌夏天 太聒噪 太闷 还没有理由睡觉 但是啊但是 天很明 水很清 云朵儿飘啊飘 不知道要悠忽到哪
    三木兔阅读 241评论 0 0
  • 最近读了两本与台湾有关的书,一本是《我们台湾这些年》,一本就是吴念真老先生的《这些人,那些事》,还有之前给我留下极...
    佐罗是我阅读 217评论 0 4