简介:redis并没有直接使用前面所提到的基本数据结构,而是基于基本的数据结构构造了一个对象系统。这个系统包含了字符串对象,列表对象,哈希对象,集合对象,有序集合对象五种类型的对象。每种对象都用到了至少一种我们前面所说的数据结构。

  • 对象的类型和编码

  redis使用对象来表示数据库中的键和值,每次当我们在redis数据库中创建一个键值对的时候,我们至少会创建两个对象。

  redis中的每个对象都由一个redisObject结构表示:

  

  1.对象类型

    对象有以下几种类型:

  

  对于redis数据库来说,键总是一个字符串对象,值可能是上述五种对象的其中一种。地方

  Type命令在面对不同类型对象时所产生的输出如下:

  

  2.对象编码

  encodeing属性决定了ptr指针指向的底层数据结构的数据结构类型也称为对象编码。

  

  每种类型的对象至少对应了两种不同的编码,下面列出每种对象可以使用的编码。

  使用object encoding命令可以查看一个数据库的值对象的编码

  

  下表展示了不同编码对象对应的object encoding命令的输出

  通过encoding属性去设置对象的编码方式,极大地提高了redis存储对象的底层数据结构的灵活性,因为不同的数据结构适用的存储场景不一样,比如列表对象包含的元素比较少的时候,redis采用压缩列表作为存储的数据结构。当元素逐渐增多到某个值的时候又会变为双端链表作为存储的数据结构。

  这是因为压缩列表比双端链表更节约内存,而且在元素数量较少时,在内存中以连续块方式保存的压缩列表比双端链表更容易载入到缓存。

  • 字符串对象

  字符串对象的编码可以是int,emstr,raw。

  int保存的整数值对象,emstr保存的是小于32字节等于32字节的对象,raw保存的是大于32字节的对象。

  emstr和raw的区别在于raw会调用两次内存分配函数分别创建redisObject结构和sdshdr结构,而embstr编码只会调用一次内存分配函数来分配一个连续的内存空间,空间中一次包含了redisObject和sdshdr两个结构。如下图所示:

  

  使用emstr编码的字符串有以下的优势:

  1》在创建和释放字符串对象时都只用对内存进行一次操作。

  2》保存在一个内存块中的对象更容易被加载到缓存。

  long和double类型的浮点数也是作为字符串保存的

  1.编码的转换

  int编码和emstr编码的字符串在条件满足的情况下可以转变为raw编码的字符串

  

  比如上述代码中类型为int的10086append一个字符串,将会使最终结果的类型变为raw类型。

  对于emstr类型的字符串只要执行了修改的命令就会将emstr类型转换为raw类型 。

  3.字符串命令的实现

  

  • 列表对象

  列表对象的编码可以是ziplist或者是linkedlist

  它们的结构如下图:

  

  

  其中linkedlist对象中的字符串是一个字符串对象,如下图所示:

  

  当列表对象同时满足以下两个条件时(不同版本的redis的定义可能不一样),列表对象将会用linkedlist实现,否则用ziplist实现。

  1》列表中所有元素的占内存数小于64字节

  2》列表中元素的个数小于512个

  

  列表命令:

  

  • 哈希对象

  哈希对象的实现可以是ziplist或者是hashtable

  两种方式分别如下图:

  

  

  当哈希对象中的元素满足以下两个条件时,哈希对象用ziplist实现,否则用hashtable实现

  1》哈希对象中的元素的占内存大小小于64字节。

  2》哈希对象中的元素的个数小于512。

  哈希命令:

  

  • 集合对象

  集合对象的实现可以是intset或者hashtable

  两种方式分别如下图:

  

  当集合对象的元素满足以下两个条件时,集合对象用intset实现,否则用hashtable实现。

  1》集合对象中的元素都是整数

  2》集合对象中的元素的数量不超过512

  集合命令:

   

   

  • 有序集合对象

  有序集合对象有ziplist和skiplist两种实现

  

  

  skiplist:

  

  

  有序集合对象满足以下两个条件的时候用ziplist实现,否则用skiplist实现。

  1》有序集合中元素的占内存大小小于64字节。

  2》有序集合中元素的个数小于128。

  有序集合命令:

  

  

redis的对象的更多相关文章

  1. redis object 对象系统

    redis object对象系统 概述 redis 当中, sds字符串, adlist双向链表, dict字典, ziplist压缩链表, intset整数集合等均为底层数据结构 redis 并没有 ...

  2. redis存储对象

      redis主要存储类型最常用的五种数据类型: String Hash List Set Sorted set redis存储对象序列化和反序列化 首先来了解一下为什么要实现序列化 为什么要实现序列 ...

  3. redis存储对象与对象序列化详解

    redis主要存储类型最常用的五种数据类型: String Hash List Set Sorted set redis存储对象序列化和反序列化 首先来了解一下为什么要实现序列化 为什么要实现序列化接 ...

  4. Redis | 使用redis存储对象反序列化异常SerializationFailedException

    案例 使用Redis进行对象存储,在处理业务逻辑的时候,丛Redis获取对象发现反序列化失败,抛出如下异常: Caused by: org.springframework.data.redis.ser ...

  5. 深入了解Redis(3)-对象

    Redis主要的数据结构有简单动态字符串(SDS).双端链表.字典.压缩列表.整数集合,等等.但Redis并没有直接使用这些数据结构来实现键值对数据库, 而是基于这些数据结构创建了一个对象系统, 这个 ...

  6. Redis 存储对象信息是用 Hash 还是 String

    Redis 内部使用一个 RedisObject 对象来表示所有的 key 和 value,RedisObject 中的 type,则是代表一个 value 对象具体是何种数据类型,它包含字符串(St ...

  7. 高性能的Redis之对象底层实现原理详解

    对象 在前面的数个章节里, 我们陆续介绍了 Redis 用到的所有主要数据结构, 比如简单动态字符串(SDS).双端链表.字典.压缩列表.整数集合, 等等. Redis 并没有直接使用这些数据结构来实 ...

  8. redis存储对象,实体类新加字段空指针问题处理

    redis是一个key-value存储系统.和Memcached类似,它支持存储的value类型相对更多,包括string(字符串).list(链表).set(集合).zset(sorted set ...

  9. redis存取对象

    redis主要存储类型最常用的五种数据类型: String Hash List Set Sorted set redis不能直接存取对象,如何解决呢? 两种方式 1.利用序列化和反序列化的方式 两层对 ...

随机推荐

  1. Java 内建函数式接口

    Java 提供一个包:java.util.function :包中主要提供四个核心接口: 1.功能性接口(Function) public interface Function<T,R> ...

  2. 【mysql】pymysql.err.InterfaceError Interface Error: (0, '')

    八成是丢失连接了 while 1: try: self.conn.ping(reconnect=True) self.cur.execute(sql,tuple(item.values())) sel ...

  3. [视频教程]利用SSH隧道进行远程腾讯云服务器项目xdebug调试

    我的远程服务器是腾讯云的ubuntu系统机器,本地我的电脑系统是deepin的系统,使用的IDE是vscode.现在就来使用本地的IDE来调试腾讯云中为网站项目实现逻辑是访问网站域名后,请求被转发给腾 ...

  4. Java面试之synchronized 和 static synchronized

    ​面试题: 答案: 不能 不能 不能 不能 能 正文 概述 通过分析这两个用法的分析,我们可以理解java中锁的概念.一个是实例锁(锁在某一个实例对象上,如果该类是单例,那么该锁也具有全局锁的概念), ...

  5. c# WF 第9节 button控件

    本节内容: 1:实现实例 1:实现实例 每当点击一个确定就出现一个窗口,当点击最后的确定时,关闭所有的窗口. 实现: 步骤1:对Form 1 -Form3 依次进行如下设置: 步骤2 : 当每点击一个 ...

  6. python多版本共存pip指向问题

    这两天一致被一个问题困扰,电脑里装了anaconda和python3.7,在命令行里输入python,想要python3.7,出现的确实python3.6,或使用pip安装包时,不知道是装在里pyth ...

  7. pindel及breandancer安装

    1.安装pindel需要依赖htslib https://github.com/samtools/htslib.git mv htslib htslib1 autoheader # If using ...

  8. 【BZOJ3600】没有人的算术(替罪羊树+线段树)

    点此看题面 大致题意: 定义任意数对\(>0\),数对之间比大小先比第一位.后比第二位,一开始数列全为\(0\),要求你支持\(a_k=(a_x,a_y)\)和询问区间最大值所在位置两种操作. ...

  9. [日常] NOI前划水日记

    NOI前划水日记 开坑记录一下每天的效率有多低 5.24 早上被春哥安排了一场NEERC(不过怎么是qualification round啊) 省队势力都跑去参加THU/PKU夏令营了...剩下四个D ...

  10. Git仓库迁移命令

    1. 从原git上clone bare下到本地 git clone --bare https://***.git 2. push mirror到目标仓库 git push --mirror https ...