Redis压缩列表的详细介绍(示例讲解)

本篇文章给大家带来的内容是关于Redis压缩列表的详细介绍(示例讲解),有一定的参考价值,有需要的朋友可以参考一下,希望对你有所帮助。

此篇文章是主要介绍Redis在数据存储方面的其中一种方式,压缩列表。

本文会介绍

1、压缩列表(ziplist)的使用场景 

2.如何达到节约内存的效果?

3.压缩列表的存储格式 

4. 连锁更新的问题  

5. conf文件配置。

在实践上的操作主要是对conf配置文件进行配置,具体上没有确切的一个值,更多是经验值。也有的项目会直接使用原本的默认值。此篇对于更好地理解一个数据库底层的存储逻辑会有一点帮助。修学储能,既要博,也要渊。希望这篇文章对同样也是在学习Redis的各位同伴有点用。(推荐教程:Redis教程)

一、压缩列表(ziplist)的使用场景:

Redis为了优化数据存储,节约内存,在列表、字典(哈希键)和有序集合的底层实现了使用压缩列表这一优化方案。

例如,假如一个哈希键里面存储的字符串比较短,那么Redis就会将它用压缩列表的格式去存储,即转换为字节数组存储。而一个哈希键内部存储的整数值比较小,同样也会把它存储为压缩列表的一个节点。同理,列表键的对小数据的存储跟哈希键的操作类似。

如此说来,压缩列表并不是开发者可以直接调用的Redis中的一种存储数据结构,而是Redis中为优化数据存储而在底层做的一项努力。理解好这点还是比较重要的。

二、如何达到节约内存的效果?

压缩列表是一种序列化的数据结构,这种数据结构的功能是将一系列数据与其编码信息存储在一块连续的内存区域,这块内存物理上是连续的。但逻辑上被分为多个组成部分,即节点。目的是为了在一定可控的时间复杂度条件下尽可能的减少不必要的内存开销,从而达到节省内存的效果。需要理解是怎么达到节约内存作用的,还需要去了解压缩列表的存储格式。

三、压缩列表的存储格式:

压缩列表(ziplist)是Redis列表键、哈希键和有序集合键的底层实现之一,其实质是一种序列化的数据存储结构。有别于普通情况下,Redis用双端链表表示列表,使用散列表表示哈希键,用散列表+跳跃表表示有序集合。当一个列表或者哈希字典/有序集合只包含很少内容,并且每一个列表项或者哈希项/有序集合项如果是小整数,或者比较短的字符串。那么Redis就会用压缩列表来做底层的实现。

压缩列表由一系列经Redis特殊编码的连续内存块组成,每一个内存块称为一个节点(entry),而一个压缩列表可以包含很多个节点。每个节点存储的数据格式可以是字节数组(中文字符串等都会转换为字节数组)或者整数值。

字节数组的长度可以是以下的其中一种:

1. 长度小于等于63字节(2的6次方)

2. 长度小于等于16383字节(2的14次方)

3. 长度小于等于4294967295字节(2的32次方)

整数值可能是以下六种中的其中一种:

1. 4位长,介于0-12之间的无符号整数

2. 1字节长的有符号整数

3. 3字节长的有符号整数

4. int16_t类型整数

5. int32_类型整数

6. int64_t类型整数

普通存储格式下和压缩列表存储格式下的不同点:

列表存储结构典型的为双端链表,每一个值都是用一个节点来表示,每个节点都会有指向前一个节点和后一个节点的指针,以及指向节点包含的字符串值的指针。而字符串值又分为3个部分存储,第一部分存储字符串长度,第二部分存储字符串值中剩余可用的字节量,第三部分存储的则是字符串数据本身。所以一个节点往往都需要存储3个指针、2个记录字符串信息的整数、字符串本省和一个额外的字节。总体上额外的开销是很大的(21字节)。

压缩列表节点的格式:

Redis压缩列表的详细介绍(示例讲解)

每一个节点都有previous_entry_length,encoding,content三个部分组成,在遍历压缩列表的时候是从后往前遍历的。

1. previous_entry_length记录了前一个节点的长度,只要用当前指针减去这个值就可以达到前一个节点的起始地址。

2. encoding记录了节点content属性所保存数据的类型和长度

Redis压缩列表的详细介绍(示例讲解)

3. content记录了一个节点的值

显然压缩列表这种方式节约了不少存储空间。但同时也会引发下面的问题。

四、连锁更新的问题:

一般而言如果前一个节点的整体长度小于254字节,previous_entry_length属性只需要1个字节的空间来保存这个长度值。而当前一个节点大于254字节的时候,previous_entry_length属性要用5个字节长的空间来记录长度值。

当长度为254字节左右的节点前插入一个新的节点的时候,需要增加previous_entry_length来记录这个节点到新节点的偏移量。这个时候,这个节点的长度肯定就大于254字节了。所以这个节点的后一个节点就不能只用一个字节的previous_entry_length来记录这个节点的信息了,而是需要5个字节来记录。如果连续多个节点的长度都为254字节左右,在其中的某一个节点前/后发生节点的插入和删除(删除的推理与插入相反,原本用5字节记录前一节点的可能变为1字节),都可能引发连锁的更新,显然,这样对系统地运行效率是很不利的。不过,在实际应用中这种情况还是比较少发生的。

而双端链表在节点的更新、增加和删除上显得就会“轻松”很多了。 因为每一个节点存储的信息都是相对独立的。  

实践意义:

爱图表 爱图表

AI驱动的智能化图表创作平台

爱图表 99 查看详情 爱图表

要预估一个节点大概占据多少字节的存储空间,适当地调整字段的存储格式而不要使存储的字段值占据存储空间落在254字节(除去encoding属性和previous_entry_length属性)左右。

Redis中查看字符串和哈希键值的长度相关命令:

1. 查询字符串键对应的值长度

命令:

Strlen

例如:

127.0.0.1:6379> strlen m_name

(integer) 8

2. 查询哈希键某一个域长度

命令:

Hstrlen

例如:

127.0.0.1:6379> hstrlen good_list good_list1

(integer) 226

五、Conf文件配置:

通过修改配置文件,可以控制是否使用压缩列表存储相关键的最大元素个数和最大元素的大小

Conf文件中的配置:
1.

[] -max-ziplist-entries : 表示对于键的最大元素个数,即一个键中在该指定值下的数量的节点个数都会用压缩列表来储存

[] -max-ziplist-value :表示压缩列表中每个节点的最大体积是多少字节

实际使用中,一个列表键/哈希键的某一个元素往往存储着比较大的信息量,会大于64字节,所以配置时很有可能会比64大,同时考虑到实际存储数据的容量大小以及上面谈到的previous_entry_length的大小问题,对[] -max-ziplist-value进行合理的配置。

配置文件内容:

############## ADVANCED CONFIG ##########################哈希键# Hashes are encoded using a memory efficient data structure when they have a# small number of entries, and the biggest entry does not exceed a given# threshold. These thresholds can be configured using the following directives.hash-max-ziplist-entries 512hash-max-ziplist-value 64有序集合键# Similarly to hashes and lists, sorted sets are also specially encoded in# order to save a lot of space. This encoding is only used when the length and# elements of a sorted set are below the following limits:zset-max-ziplist-entries 128zset-max-ziplist-value 64列表键,比较特殊,直接使用制定大小kb字节数表示(有些conf文件的列表键与hash键的表达式没太大区别)# Lists are also encoded in a special way to save a lot of space.# The number of entries allowed per internal list node can be specified# as a fixed maximum size or a maximum number of elements.# For a fixed maximum size, use -5 through -1, meaning:# -5: max size: 64 Kb  <-- not recommended for normal workloads# -4: max size: 32 Kb  <-- not recommended# -3: max size: 16 Kb  <-- probably not recommended# -2: max size: 8 Kb   <-- good# -1: max size: 4 Kb   <-- good# Positive numbers mean store up to _exactly_ that number of elements# per list node.# The highest performing option is usually -2 (8 Kb size) or -1 (4 Kb size),# but if your use case is unique, adjust the settings as necessary.list-max-ziplist-size -2

案例:

修改配置前使用默认配置:

hash-max-ziplist-entries 512

hash-max-ziplist-value 64

127.0.0.1:6379> hstrlen good_list good_list1

(integer) 226

127.0.0.1:6379> object encoding good_list

“hashtable”

修改配置:

hash-max-ziplist-entries 512

hash-max-ziplist-value 254

注意:修改配置后需要重启服务器

127.0.0.1:6379> hstrlen good_list good_list1

(integer) 226

127.0.0.1:6379> object encoding good_list

“ziplist”

可以看到存储方式已将变为ziplist

较官方的压力测试和指导建议:

当一个压缩列表的元素数量上升到几千(实际使用可能远小于这个值)的时候,压缩列表的性能可能会下降,因为Redis在操作这种结构的时候,编解码会出现一定的压力。

压缩列表的长度限制在500-2000之内,每个元素体积限制在128字节或以下,压缩列表的的性能都会处于合理范围之内。

以上就是Redis压缩列表的详细介绍(示例讲解)的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/376789.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年11月6日 09:54:13
下一篇 2025年11月6日 09:58:54

相关推荐

  • php数据库如何实现数据缓存 php数据库减少查询压力的方案

    答案:PHP结合Redis等内存缓存系统可显著提升Web应用性能。通过将用户信息、热门数据等写入内存缓存并设置TTL,先查缓存未命中再查数据库,减少数据库压力;配合OPcache提升脚本执行效率,文件缓存适用于小型项目,数据库缓冲池优化和读写分离进一步提升性能,推荐Redis为主并防范缓存穿透与雪崩…

    2025年12月6日 后端开发
    000
  • win11提示“无法加载文件或程序集”错误怎么办_Win11文件或程序集无法加载修复方法

    首先运行SFC扫描修复系统文件,若无效则使用DISM修复系统映像,接着重装Visual C++运行库,再通过设置修复或重置出错应用,最后可手动替换并注册缺失的程序集文件。 如果您尝试在Windows 11系统中运行某个程序或打开文件时,收到“无法加载文件或程序集”的错误提示,则可能是由于系统文件损坏…

    2025年12月6日
    000
  • 如何在安装完成后优化缓存使用

    合理配置缓存策略可提升系统效率,需设置适宜的过期时间、选用多层存储介质并持续监控维护。 安装完成后优化缓存使用,关键在于合理配置缓存策略、选择合适的存储方式,并定期维护。以下是一些实用建议,帮助你提升系统或应用的缓存效率。 合理设置缓存过期时间 缓存的有效期直接影响数据的新鲜度和性能表现。设置过短会…

    2025年12月6日 数据库
    000
  • thinkphp队列任务不执行怎么解决

    答案是检查配置、推送、监听和执行环节。首先确认queue.php中驱动设置正确,如Redis或数据库配置无误;其次确保任务通过Queue::push正确推送到队列,并在对应存储查看记录;然后必须运行php think queue:work –daemon命令启动监听;最后检查fire方法…

    2025年12月6日 PHP框架
    000
  • PHP框架怎么使用缓存机制_PHP框架缓存驱动配置与数据缓存策略

    答案:现代PHP开发中,缓存通过减少数据库压力提升性能,主流框架如Laravel支持File、Redis、Memcached、Database等驱动,推荐生产环境使用Redis;合理设置过期时间、规范缓存键命名、条件性更新及分层缓存策略可优化性能,结合Cache::remember等方法实现高效数据…

    2025年12月6日 后端开发
    000
  • 使用PHP框架开发实时应用_基于Laravel的php框架怎么用的通信

    Laravel通过事件广播与WebSocket实现实时通信,1. 配置Redis或Pusher驱动并创建实现ShouldBroadcast接口的事件;2. 使用laravel/websockets扩展启动WebSocket服务器;3. 前端引入Laravel Echo连接本地WebSocket并监听…

    2025年12月6日 后端开发
    000
  • 如何在Laravel中实现缓存机制

    laravel的缓存机制用于提升应用性能,通过存储耗时操作结果避免重复计算。1. 配置缓存驱动:在.env文件中设置cache_driver,如redis,并安装相应扩展;2. 使用cache facade进行缓存操作,包括put、get、has、forget等方法;3. 使用remember和pu…

    2025年12月5日
    000
  • win8打开程序提示0xc000007b怎么办_win8程序0xc000007b错误解决方法

    首先重新安装Visual C++ Redistributable运行库,包括x86和x64版本;其次修复DirectX组件,更新至最新运行时;然后运行SFC扫描修复系统文件;最后手动注册vcruntime140.dll等关键DLL文件,每步完成后重启电脑测试程序。 如果您在Windows 8系统中尝…

    2025年12月5日
    000
  • Serverless架构下Workerman的无状态化改造方案

    在serverless架构下,workerman的无状态化改造可以通过以下步骤实现:1. 将workerman的逻辑拆分成独立的函数,如handleconnect、handlemessage和handleclose。2. 使用外部服务(如redis或dynamodb)存储状态信息。3. 采用事件驱动…

    2025年12月5日
    000
  • 应用程序发生异常0xc0000417怎么解决 5种解决方案揭晓

    方法一:重启系统并清理后台占用程序 有时候,出现“0xc0000417”错误只是因为系统临时资源冲突,尤其是在同时运行多个程序或后台任务过多的情况下。 1、先保存好当前的工作内容,然后重启计算机。 2、使用快捷键 Ctrl + Shift + Esc 调出任务管理器,查看正在运行的进程,关闭不必要的…

    2025年12月5日 电脑教程
    000
  • 电脑的dll文件丢失怎么恢复 一文教你快速恢复

    在使用计算机的过程中,不少用户可能会碰到类似的问题:启动某个程序或游戏时,系统弹出提示信息,例如“xxx.dll文件缺失”或“程序无法运行”。这类问题通常是由dll(动态链接库)文件被误删、损坏或丢失所引起的。本文将为你整理一套完整的应对策略,帮助你迅速找回丢失的dll文件,恢复正常系统运行。 一、…

    2025年12月5日 电脑教程
    000
  • 找不到Qt5Widgets.dll无法执行代码 解决方法推荐

    电脑在运行某些基于qt框架开发的应用程序时,可能会弹出提示:“无法找到qt5widgets.dll,代码执行无法继续。重新安装程序可能解决此问题。”这类错误通常由程序安装不完整、系统中关键dll文件丢失或环境变量配置异常引起,尤其在使用开源软件、跨平台工具或图形化界面应用时较为常见。本文将从三个实用…

    2025年12月5日 电脑教程
    000
  • 如何在Laravel中配置Redis缓存

    在laravel中配置redis缓存的核心步骤包括安装并运行redis服务、安装php扩展或composer包、配置.env文件和config/database.php、清除缓存。1. 安装redis服务器:使用系统包管理工具安装并启动redis服务;2. 安装php扩展或predis包:选择php…

    2025年12月5日
    000
  • 如何在Laravel中配置邮件发送服务

    laravel中配置邮件发送服务的核心是利用其邮件抽象层,通过修改.env文件和config/mail.php对接各种邮件服务商。1. 配置.env文件设置mail_mailer、mail_host、mail_port、mail_username、mail_password、mail_encrypt…

    2025年12月5日
    100
  • Java中如何实现限流 掌握流量控制

    在java中实现限流的方法主要包括计数器算法、滑动窗口算法、漏桶算法、令牌桶算法以及使用guava ratelimiter。1. 计数器算法通过设定时间窗口和请求数量进行限制,优点是实现简单,缺点是可能存在“突刺”问题;2. 滑动窗口算法将时间窗口细化,避免了“突刺”,效果更平滑但实现较复杂;3. …

    2025年12月5日 java
    000
  • PHP如何调用Scala代码 通过JVM桥接调用Scala程序的方法

    通过jvm桥接,php可调用scala代码,但需中间工具。具体步骤如下:1. 将scala代码编译为jar包,并确保类和方法为public;2. 部署javabridge到支持servlet的web服务器(如tomcat);3. 在php中配置java.inc并设置classpath以加载jar包;…

    2025年12月5日 后端开发
    000
  • 如何在Laravel中使用事件和监听器

    事件和监听器是laravel中实现松耦合的关键机制。1. 定义事件类如userregistered,封装发生的“事情”;2. 创建监听器如sendwelcomeemail,处理事件触发后的操作,并可异步执行;3. 在eventserviceprovider中注册事件与监听器的映射关系;4. 使用ev…

    2025年12月5日
    000
  • 如何在Laravel中配置队列工作器

    在laravel中配置队列工作器的核心步骤是设置队列驱动并启动监听进程,以提升应用性能和用户体验。1. 修改.env文件中的queue_connection变量,如设为redis以启用高性能队列;2. 配置redis连接信息确保其可用性;3. 使用php artisan queue:work命令启动…

    2025年12月5日
    100
  • 如何在Laravel中实现数据补全

    数据补全功能可在用户输入不完整信息时智能猜测并提供完整选项,laravel中可通过前后端协作实现:1.前端使用typeahead.js、select2或awesomplete监听输入框并发送ajax请求;2.后端创建路由和控制器接收输入值,用eloquent orm结合like或全文搜索技术查询数据…

    2025年12月5日
    000
  • 如何在Laravel中实现定时任务

    在laravel中实现定时任务,核心思路是利用框架的调度器集中管理任务,并通过服务器cron每分钟触发一次调度器执行。1. 创建命令:使用 php artisan make:command 生成命令类并编写业务逻辑;2. 注册任务:在 app/console/kernel.php 的 schedul…

    2025年12月5日
    000

发表回复

登录后才能评论
关注微信