前言

参考资料:《Redis设计与实现 第二版》;

本篇笔记按照书里的脉络,将知识点分为四个部分。其中第一部分数据结构与对象分为上中下篇,上篇包括:SDS链表字典;中篇包括跳跃表整数集合压缩列表;下篇为对象

上篇的链接:https://www.cnblogs.com/dlhjw/p/15569578.html

中篇的链接:https://www.cnblogs.com/dlhjw/p/15582039.html

Redis常用命令及示例总结:https://blog.csdn.net/dlhjw1412/article/details/119713214


1. Redis对象概述

  • Redis没有直接使用前面介绍的数据结构来实现键值对数据库,而是基于这些数据结构的对象系统;

  • Redis有五种基本对象,分别是字符串列表哈希集合有序集合。使用以下8种编码方式中的几种作为底层实现底层实现:long类型整数embstr编码的SDSSDS字典双端链表压缩列表整数集合跳跃表

  • Redis在创建键值对时,至少会生成两个对象,键对象和值对象;

1.1 对象的定义

  • Redis中的每个对象都由一个redisObject结构来表示:

    typedef struct redisObject{
    //类型
    unsigned type:4;
    //编码
    unsigned encoding:4;
    //指向底层实现数据结构的的指针
    void *ptr;
    //……
    }
    • 类型type的可选类型:

      类型常量 对象的名称 TYPE命令的输出
      REDIS_STRING 字符串对象 string
      REDIS_LIST 列表对象 list
      REDIS_HASH 哈希对象 hash
      REDIS_SET 集合对象 set
      REDIS_ZSET 有序集合对象 zset
    • 编码encoding的可选类型;同种类型可以有不同的编码形式:

      类型 对象 编码 编码方式 OBJECT ENCODING命令输出
      REDIS_STRING 字符串 REDIS_ENCODING_INT 使用整数数值实现 int
      REDIS_STRING 字符串 REDIS_ENCODING_EMBSTR 使用embstr编码 embstr
      REDIS_STRING 字符串 REDIS_ENCODING_RAW 使用SDS实现 raw
      REDIS_LIST 列表 REDIS_ENCODING_ZIPLIST 使用压缩列表实现 ziplist
      REDIS_LIST 列表 REDIS_ENCODING_LINKEDLIST 使用双端链表实现 linkedlist
      REDIS_HASH 哈希 REDIS_ENCODING_ZIPLIST 使用压缩列表实现 ziplist
      REDIS_HASH 哈希 REDIS_ENCODING_HT 使用字典实现 hashtable
      REDIS_SET 集合 REDIS_ENCODING_INTSET 使用整数集合实现 intset
      REDIS_SET 集合 REDIS_ENCODING_HT 使用字典实现 hashtable
      REDIS_ZSET 有序集合 REDIS_ENCODING_ZIPLIST 使用压缩列表实现 ziplist
      REDIS_ZSET 有序集合 REDIS_ENCODING_SKIPLIST 使用跳跃表字典实现 skiplist

2. 字符串对象

  • 字符串编码可以是intrawembstr

    编码类型 说明
    int 字符串保存整数值,并且这个整数可以用long类型表示
    raw 字符串值的长度大于39字节
    embstr 字符串值的长度小于39字节
  • embstr编码是专门用于保存短字符串的一种优化编码方式;

  • rawembstr的异同:

    • 二者都使用redisObject结构与sdshdr结构来表示字符串;
    • embstr通过调用一次内存分配函数来分配一块连续的空间;
    • raw通过调用两次内存分配函数来分配一块连续的空间;
  • embstr的优点:

    • 创建时只需要分配一次内存;
    • 释放时只需要调用一次内存释放函数;
    • 连续保存在一块连续内存里,对缓存友好;

  • long double类型表示的浮点数在Redis中也是作为字符串值来保存的;

  • embstr编码的字符串对象实际上是只读的,要修改先会转成raw编码,再执行修改命令;

  • 字符串命令请见《Redis常用命令及示例总结》

3. 列表对象

  • 列表的编码对象可以是ziplistlinkedlist
  • redis 3.2以后,quicklist作为列表键的实现底层实现之一,代替了压缩列表。
  • ziplist编码的条件:
    • 列表对象保存的所有字符串元素的长度都小于64字节;
    • 列表对象保存的元素数量小于512个;

3.1 quicklist 快速链表

  • quicklist的定义在quicklist.h

    typedef struct quicklist {
    //指向头部(最左边)quicklist节点的指针
    quicklistNode *head;
    //指向尾部(最右边)quicklist节点的指针
    quicklistNode *tail;
    //ziplist中的entry节点计数器
    unsigned long count;
    //quicklist的quicklistNode节点计数器
    unsigned int len;
    //保存ziplist的大小,配置文件设定,占16bits
    int fill : 16;
    //保存压缩程度值,配置文件设定,占16bits,0表示不压缩
    unsigned int compress : 16;
    } quicklist;
  • quicklist节点的定义:

    typedef struct quicklistNode {
    struct quicklistNode *prev; //前驱节点指针
    struct quicklistNode *next; //后继节点指针
    //不设置压缩数据参数recompress时指向一个ziplist结构
    //设置压缩数据参数recompress指向quicklistLZF结构
    unsigned char *zl;
    //压缩列表ziplist的总长度
    unsigned int sz;
    //ziplist中包的节点数,占16 bits长度
    unsigned int count : 16;
    //表示是否采用了LZF压缩算法压缩quicklist节点,1表示压缩过,2表示没压缩,占2 bits长度
    unsigned int encoding : 2;
    //表示一个quicklistNode节点是否采用ziplist结构保存数据,2表示压缩了,1表示没压缩,默认是2,占2bits长度
    unsigned int container : 2;
    //标记quicklist节点的ziplist之前是否被解压缩过,占1bit长度
    //如果recompress为1,则等待被再次压缩
    unsigned int recompress : 1;
    //测试时使用
    unsigned int attempted_compress : 1;
    //额外扩展位,占10bits长度
    unsigned int extra : 10;
    } quicklistNode;

4. 哈希对象

  • 哈希对象编码可以是ziplisthashtable

  • 使用ziplist编码的条件:

    • 键和值的字符串长度小于64字节;
    • 键值对数量少于512个;
  • 使用ziplist编码时:

  • 使用hashtable编码时:

5. 集合对象

  • 集合对象的编码可以是intsethashtable
  • 使用inset编码的条件:
    • 所有元素为整数值;
    • 保存的元素不超过512个;
  • 使用inset编码时:

  • 使用hashtable编码时:

  • 集合命令请见《Redis常用命令及示例总结》

6. 有序集合对象

  • 有序集合的编码可以是ziplistskiplist

  • 压缩列表内的集合元素按分值从小到大排列;

  • 使用ziplist编码时:



  • 使用ziplist编码的条件:

    • 元素数量少于128个;
    • 元素长度小于64字节;
  • 使用skiplist编码时,使用zset结构作为底层实现;

  • zset的定义:

    typedef struct zset{
    //跳跃表
    zskiplist *zsl;
    //字典
    dict *dict;
    } zset;
    • 跳跃表和字典使用指针来共享相同的元素和分值,因此不会产生重复,也不会造成内存浪费;

  • 有序集合命令请见《Redis常用命令及示例总结》

7. Redis对象的特点

7.1 类型检查与命令多态

  • Redis的命令基本上分两类。一种是可以对任意类型的键操作(基于类型的多态),一种是只能对特定类型的键执行(基于编码的多态);
  • 在执行特定类型命令之前,服务器会先检查redisObject结构的type属性,判断是否为执行该命令所需的类型。是则执行,否则返回类型错误;
  • Redis还会根据值对象的编码方式选择正确底层方法,使一个命令可以同时用于处理多种不同编码方式的数据结构,进而实现多态命令;

7.2 内存回收

  • C语言不具备自动回收内存的功能,Redis构建一个引用计数计数实现内存回收机制;

  • 引用计数由redisObject结构的refcount属性记录:

    typedef struct redisObject{
    //...
    //引用计数
    int refcount;
    //...
    }
    • 创建新对象时,refcount被初始化为1;
    • 对象被新程序使用时,refcount++
    • 对象不被一个程序使用时,refcount--
    • 对象计数值为0时,对象占用的内存会被释放;

7.3 对象共享

  • 对象的计数属性带有对象共享的作用;
  • 当多个键保存同一个值时,这些键的值指针指向同一个值对象,值对象的refcounr为n;
  • Redis在初始化服务器时,会创建一万个字符串对象(0~9999的字符串对象),当服务器需要用到这些值对象时,服务器会使用这些共享对象,而不是创建新对象;
  • Redis只对包含整数值的字符串对象进行共享,验证数字的时间复杂度为O(1);

7.4 对象的空转时长

  • redisObject结构里有个lru属性,记录对象最后一次被命令程序访问的时间;
    typedef struct redisObject{
    //...
    unsigned lru:22;
    //...
    }
  • 使用命令OBJECT IDLETIME可以显示对象的空转时间,不会改变对象的空转时间;
  • 如果服务器打开maxmemory选项,并且回收内存的算法为volatile-lruallkeys-lru,那么当服务器占用的内存数超过maxmemory选项设置的上限值是,空转时间较高的键会优先被服务器释放,回收内存;

最后

新人制作,如有错误,欢迎指出,感激不尽!
欢迎关注公众号,会分享一些更日常的东西!
如需转载,请标注出处!

Redis | 第一部分:数据结构与对象 下篇《Redis设计与实现》的更多相关文章

  1. [redis读书笔记] 第一部分 数据结构与对象 对象类型

    - 从前面redis的基本数据结构来看,可以看出,redis都是在基本结构(string)的基础上,封装了一层统计的结构(SDS),这样让对基本结构的访问能够更快更准确,提高可控制度. - redis ...

  2. Redis笔记(1)数据结构与对象

    1.前言 此系列博客记录redis设计与实现一书的笔记,提取书本中的知识点,省略相关说明,方便查阅. 2.基本数据结构 2.1 简单动态字符串SDS(simple dynamic string) 结构 ...

  3. Redis 的底层数据结构(对象)

    目前为止,我们介绍了 redis 中非常典型的五种数据结构,从 SDS 到 压缩列表,这都是 redis 最底层.最常用的数据结构,相信你也掌握的不错. 但 redis 实际存储键值对的时候,是基于对 ...

  4. 左手Mongodb右手Redis 第一章,进入Mongodb和Redis的世界

    ---恢复内容开始--- 1,为什么要使用非关系型数据库,关系型数据库咋滴,不能用嘛? 存在即合理,非关系型数据库的出现,那说明关系型数据库不适用了. 非关系型数据库(NOSQL)-->Not ...

  5. .net core工具组件系列之Redis—— 第一篇:Windows环境配置Redis(5.x以上版本)以及部署为Windows服务

    Cygwin工具编译Redis Redis6.x版本是未编译版本(官方很调皮,所以没办法,咱只好帮他们编译一下了),所以咱们先下载一个Cygwin,用它来对Redis进行编译. Cygwin下载地址: ...

  6. [REDIS 读书笔记]第一部分 数据结构与对象 跳跃表

    下面是跳跃表的基本原理,REDIS的实现大致相同 跳跃表的一个特点是,插入NODE是通过随机的方式来决定level的,比较奇特 下面是skipList的一个介绍,转载来的,源地址:http://ken ...

  7. [redis读书笔记] 第一部分 数据结构与对象 对象特性

    一 类型检查和多态    类型检查,即有的命令是只针对特定类型的,如果类型不对,就会报错,此处的类型,是指的键类型,即robj.type.下面为有类型检查的命令: 对于某一种类型,redis下底层的实 ...

  8. [redis读书笔记] 第一部分 数据结构与对象 字典

    三 字典 字典是Hash对象的底层实现,比如用HSET创建一个HASH的对象,底层可能就是用一个字典实现的键值对. 字典的实现主要设计下面三个结构: /* * 哈希表节点 */ typedef str ...

  9. [redis读书笔记] 第一部分 数据结构与对象 简单动态字符串

    本读书笔记主要来自于<<redis设计与实现>> -- 黄键宏(huangz) redis主要设计了字符串,链表,字典,跳跃表,整数集合,压缩列表来做为基本的数据结构,实现键值 ...

随机推荐

  1. 洛谷2900 [USACO08MAR]土地征用Land Acquisition (斜率优化+dp)

    自闭的一批....为什么斜率优化能这么自闭. 首先看到这个题的第一想法一定是按照一个维度进行排序. 那我们不妨直接按照\(h_i\)排序. 我们令\(dp[i]\)表示到了第\(i\)个矩形的答案是多 ...

  2. ToString()字符串转换你用正确了吗?

    前言 在开发中,ToString()这个方法太方便了,以致于误解大家转换时都是直接Object.ToString()直接转换, 其实不然, 一般都是转之前先判断是否为null后再进行转换,否则会直接报 ...

  3. Java(31)泛型和可变参数

    作者:季沐测试笔记 原文地址:https://www.cnblogs.com/testero/p/15228443.html 博客主页:https://www.cnblogs.com/testero ...

  4. bash反弹shell

    part1:不求甚解的本地复现 攻击端Debian 10.x:  192.168.208.134 受害端Ubuntu : 192.168.208.135 攻击端打开(监听)某端口:  键入命令:[nc ...

  5. 【UE4 C++】资源烘焙与UE4Editor.exe启动

    资源烘焙 虚幻引擎以内部使用的特定格式存储内容资源,将内容从内部格式转换为特定于平台的格式的过程 称为 烘焙((Cooking) 从编辑器烘焙资源 FIle → Cook Content for Wi ...

  6. Spring Cloud Gateway GatewayFilter的使用

    Spring Cloud Gateway GatewayFilter的使用 一.GatewayFilter的作用 二.Spring Cloud Gateway内置的 GatewayFilter 1.A ...

  7. 使用flink实现一个简单的wordcount

    使用flink实现一个简单的wordcount 一.背景 二.需求 三.前置条件 1.jdk版本要求 2.maven版本要求 四.实现步骤 1.创建 flink 项目 2.编写程序步骤 1.创建Str ...

  8. 15个问题自查你真的了解java编译优化吗?

    摘要:为什么C++的编译速度会比java慢很多?二者运行程序的速度差异在哪? 了解了java的早期和晚期过程,就能理解这个问题了. 本文分享自华为云社区<你真的了解java编译优化吗?15个问题 ...

  9. 利用Nginx搭建Ambari本地安装源

    1.下载本地源包https://docs.hortonworks.com/HDPDocuments/Ambari-2.7.3.0/bk_ambari-installation/content/ch_o ...

  10. 奔跑吧linux-第三章实验

    基于树莓派+openeuler平台 实验 3-2:汇编语言练习--查找最大数 1.实验目的 通过本实验了解和熟悉 ARM64 汇编语言. 2.实验要求 使用 ARM64 汇编语言来实现如下功能:在给定 ...