[TOC]html
引用:Redis详解(四)------ redis的底层数据结构redis
判断值类型:
object encoding keyname
算法
127.0.0.1:6379> set k1 str
OK
127.0.0.1:6379> set k2 123
OK
127.0.0.1:6379> Object encoding k1
"embstr"
127.0.0.1:6379> Object encoding k2
"int"
127.0.0.1:6379> lpush list1 1 2 3
(integer) 3
127.0.0.1:6379> Object encoding list1
"quicklist"
127.0.0.1:6379> hset testhash key1 009
(integer) 1
127.0.0.1:6379> object encoding testhash
"ziplist"
127.0.0.1:6379> hset testhash key2 00ososkskkalalkskskaaakasmasd,jansm,namsnasnda,msn,akdwj,kjallllllaaaaassjjjjjjjjacacascascascmnascaksc,ascmascna,klna,cksa,cksans,can,scamcs9
(integer) 1
127.0.0.1:6379> object encoding testhash
"hashtable"
127.0.0.1:6379>
复制代码
数据结构数据库
数据结构 | 定义常量 | 值 |
---|---|---|
整数类型 | REDIS_ENCODING_INT | "int" |
embstr字符串类型 | REDIS_ENCODING_EMBSTR | "embstr" |
简单动态字符串 | REDIS_ENCODING_RAW | "raw" |
字典类型 | REDIS_ENCODING_HT | "hashtable" |
双端链表 | REDIS_ENCODING_LINKEDLIST | "linkedlist" |
压缩列表 | REDIS_ENCODING_ZIPLIST | "ziplist" |
整数集合 | REDIS_ENCODING_INTSET | "intset" |
跳表和字典 | REDIS_ENCODING_SKIPLIST | "skiplist" |
对应关系数组
1. string ==> raw|embstr|int
2. list ==> quicklist
3. hash ==> ziplist|hashtable
4. set ==> intset|hashtable
5. zset ==> ziplist|skiplist
6. stream => stream
复制代码
struct sdshdr{
//记录buf数组中已使用字节的数量
//等于 SDS 保存字符串的长度
int len;
//记录 buf 数组中未使用字节的数量
int free;
//字节数组,用于保存字符串
char buf[];
}
复制代码
SDS保存的字符串结构图示: 安全
常数复杂度获取字符串长度bash
杜绝缓冲区溢出数据结构
减小修改字符串的内存从新分配次数函数
二进制安全ui
兼容部分C字符串函数
//链表节点
typedef struct listNode{
//前置节点
struct listNode *prev;
//后置节点
struct listNode *next;
//节点的值
void *value;
}listNode
复制代码
经过多个 listNode 结构就能够组成链表,这是一个双端链表,Redis还提供了操做链表的数据结构:
typedef struct list{
//表头节点
listNode *head;
//表尾节点
listNode *tail;
//链表所包含的节点数量
unsigned long len;
//节点值复制函数
void (*free) (void *ptr);
//节点值释放函数
void (*free) (void *ptr);
//节点值对比函数
int (*match) (void *ptr,void *key);
}list;
复制代码
字典又称为符号表或者关联数组、或映射(map),是一种用于保存键值对的抽象数据结构。字典中的每个键 key 都是惟一的,经过 key 能够对值来进行查找或修改。 - - Redis 的字典使用哈希表做为底层实现
typedef struct dictht{
//哈希表数组
dictEntry **table;
//哈希表大小
unsigned long size;
//哈希表大小掩码,用于计算索引值
//老是等于 size-1
unsigned long sizemask;
//该哈希表已有节点的数量
unsigned long used;
}dictht
复制代码
哈希表是由数组 table 组成,table 中每一个元素都是指向 dict.h/dictEntry 结构,dictEntry 结构定义以下:
typedef struct dictEntry{
//键
void *key;
//值
union{
void *val;
uint64_tu64;
int64_ts64;
}v;
//指向下一个哈希表节点,造成链表
struct dictEntry *next;
}dictEntry
复制代码
key 用来保存键,val 属性用来保存值,值能够是一个指针,也能够是uint64_t整数,也能够是int64_t整数。
注意这里还有一个指向下一个哈希表节点的指针,咱们知道哈希表最大的问题是存在哈希冲突,如何解决哈希冲突,有开放地址法和链地址法。这里采用的即是链地址法,经过next这个指针能够将多个哈希值相同的键值对链接在一块儿,用来解决哈希冲突。
#一、使用字典设置的哈希函数,计算键 key 的哈希值
hash = dict->type->hashFunction(key);
#二、使用哈希表的sizemask属性和第一步获得的哈希值,计算索引值
index = hash & dict->ht[x].sizemask;
复制代码
跳跃表(skiplist)是一种有序数据结构,它经过在每一个节点中维持多个指向其它节点的指针,从而达到快速访问节点的目的。
//表节点定义
typedef struct zskiplistNode {
//层
struct zskiplistLevel{
//前进指针
struct zskiplistNode *forward;
//跨度
unsigned int span;
}level[];
//后退指针
struct zskiplistNode *backward;
//分值
double score;
//成员对象
robj *obj;
} zskiplistNode
复制代码
多个跳跃表节点构成了一个跳跃表
typedef struct zskiplist{
//表头节点和表尾节点
structz skiplistNode *header, *tail;
//表中节点的数量
unsigned long length;
//表中层数最大的节点的层数
int level;
}zskiplist;
复制代码
//列表节点
typedef struct ziplistNode{
// 记录压缩列表前一个字节的长度.
int previous_entry_length;
// 节点的content的内容类型以及长度.encoding类型一共有两种,一种字节数组一种是整数,encoding区域长度为1字节、2字节或者5字节长。
buf encoding;
// 节点的内容,节点内容类型和长度由encoding决定。
buf content;
}
// 压缩表
typedef struct ziplist{
//表头节点和表尾节点
structz ziplistNode entryX;
//表中节点的数量
unsigned long length;
//表中层数最大的节点的层数
int zlbytes;
int zltail;
int zlen;
int zlend:
}ziplist;
复制代码
结构图
1.保存的元素数量小于128;
2.保存的全部元素长度都小于64字节。
不能知足上面两个条件的使用 skiplist 编码。以上两个条件也能够经过Redis配置文件zset-max-ziplist-entries 选项和 zset-max-ziplist-value 进行修改。
整数集合intset是Redis用于保存整数值的集合抽象数据类型,他能够保存类型为int16_t,int32_t或者int64_t的整数值,而且保证集合中不会出现重复元素。
typedef struct intset{
//编码方式
uint32_t encoding;
//集合包含的元素数量
uint32_t length;
//保存元素的数组
int8_t contents[];
}intset;
复制代码
整数集合的每个元素都是contents数组的一个数据项,他们按照从大到小的顺序排列,而且不包含任何重复项 length属性记录来contents数组的大小 须要注意的是虽然contents数组声明为int_8类型,可是实际上contents数组并不保存任何int_8类型的值,其真正类型由encoding来决定。