Redis数据结构(一)简单动态字符串

本文涉及的产品
云数据库 Tair(兼容Redis),内存型 2GB
Redis 开源版,标准版 2GB
推荐场景:
搭建游戏排行榜
简介: Redis的字符串采用的是自定义的struct,名字叫做简单动态字符串(simple dynamic string,SDS)。 结构如下:struct sdshdr{int len;int free;char buf[];};采用如此结构的好处是: 【1】获取length的时候复杂度为O(1),不需要O(n); 【2】动态分配空间,避免缓

Redis的字符串采用的是自定义的struct,名字叫做简单动态字符串(simple dynamic string,SDS)。
结构如下:

struct sdshdr{
int len;
int free;
char buf[];
};

采用如此结构的好处是:
【1】获取length的时候复杂度为O(1),不需要O(n);
【2】动态分配空间,避免缓冲区溢出,避免每次修改或者append都重新分配;
【3】二进制安全;
关于第一点显而易见,第二点,为了减少修改字符串带来的内存重分配次数,redis采用了2个措施:
1)空间预分配;假设如下sds(状态【0】),执行命令【sdscat(s,”redis”)】往其中append一个字符串“redis”之后将变为状态【1】,发生了什么呢?当sds发现当前free长度无法分配新添加的字符串时,将发生一次空间分配,如果修改之后sds长度小于1MB,则会分配总长度为【修改后长度】*2+1,即为(5+5)*2+1,1为结尾符号空间。如果修改后sds总长度大于等于1MB,则会分配【修改后总长度】+1MB的长度。假设再次执行命令【sdscat(s,”redis”)】,由于当前状态【1】free=10,有足够空间分配新加入的字符串,则不会发生空间分配操作,变为状态【2】;

状态【0】

   struct sdshdr{
   int len=5;
   int free=4;
   char buf[]={'h','e','l','l','o','\0','','','',''};
   }; 

状态【1】:

   struct sdshdr{
   int len=10;
   int free=10;
   char buf[]={'h','e','l','l','o','r','e','d','i','s','\0',...};
   }; 

状态【2】:

   struct sdshdr{
   int len=15;
   int free=5;
   char buf[]={'h','e','l','l','o','r','e','d','i','s','r','e','d','i','s','\0',...};
   }; 

2)惰性空间释放;现在假设从状态【2】执行命令【sdstrim(s,”re”)】,移除sds中所有的’r’,’e’,将转换为状态【3】; 此时并没有释放空间,len+free依然等于20;这样做的目的是避免了缩短字符串时的内存重分配操作,并且未将来的字符串扩充预留了空间 ; 当然也可以使用【sdsfree】真正释放sds;
状态【3】:

   struct sdshdr{
   int len=11;
   int free=9;
   char buf[]={'h','e','l','l','o','d','i','s','d','i','s','\0',...};
   }; 

关于第三点,SDS的buf属性称为字节数组,保存的是二进制数据;同时由于使用len字段来判断字符串是否结束,所以是安全的,不会出现c字符串中以空格作为结尾判断,导致字符串被截断的问题。

相关实践学习
基于Redis实现在线游戏积分排行榜
本场景将介绍如何基于Redis数据库实现在线游戏中的游戏玩家积分排行榜功能。
云数据库 Redis 版使用教程
云数据库Redis版是兼容Redis协议标准的、提供持久化的内存数据库服务,基于高可靠双机热备架构及可无缝扩展的集群架构,满足高读写性能场景及容量需弹性变配的业务需求。 产品详情:https://www.aliyun.com/product/kvstore     ------------------------------------------------------------------------- 阿里云数据库体验:数据库上云实战 开发者云会免费提供一台带自建MySQL的源数据库 ECS 实例和一台目标数据库 RDS实例。跟着指引,您可以一步步实现将ECS自建数据库迁移到目标数据库RDS。 点击下方链接,领取免费ECS&RDS资源,30分钟完成数据库上云实战!https://developer.aliyun.com/adc/scenario/51eefbd1894e42f6bb9acacadd3f9121?spm=a2c6h.13788135.J_3257954370.9.4ba85f24utseFl
目录
相关文章
|
1天前
|
NoSQL Redis
Redis 字符串(String)
10月更文挑战第16天
13 4
|
13天前
|
消息中间件 存储 缓存
redis支持的数据结构
redis支持的数据结构
22 2
|
27天前
|
存储 缓存 NoSQL
3)深度解密 Redis 的字符串
3)深度解密 Redis 的字符串
26 1
|
1月前
|
存储 JSON NoSQL
redis基本数据结构(String,Hash,Set,List,SortedSet)【学习笔记】
这篇文章是关于Redis基本数据结构的学习笔记,包括了String、Hash、Set、List和SortedSet的介绍和常用命令。文章解释了每种数据结构的特点和使用场景,并通过命令示例演示了如何在Redis中操作这些数据结构。此外,还提供了一些练习示例,帮助读者更好地理解和应用这些数据结构。
redis基本数据结构(String,Hash,Set,List,SortedSet)【学习笔记】
|
17天前
|
NoSQL Redis C++
Redis的实现五:二叉堆的数据结构和TTL、c,c++的实现
这篇文章详细探讨了二叉堆的数据结构及其在C和C++中的实现,特别强调了二叉堆在Redis中实现TTL(生存时间)功能的重要性,并通过代码示例展示了如何在Redis中使用二叉堆来管理键的过期时间。
28 0
|
2月前
|
C# 开发者 UED
WPF开发者必备秘籍:深度解析文件对话框使用技巧,打开与保存文件原来如此简单!
【8月更文挑战第31天】在WPF应用开发中,文件操作是常见需求。本文详细介绍了如何利用`Microsoft.Win32`命名空间下的`OpenFileDialog`和`SaveFileDialog`类来正确实现文件打开与保存功能。通过示例代码展示了如何设置文件过滤器、初始目录等属性,并使用对话框进行文件读写操作。正确使用文件对话框能显著提升用户体验,使应用更友好易用。
63 0
|
11天前
|
算法 程序员 索引
数据结构与算法学习七:栈、数组模拟栈、单链表模拟栈、栈应用实例 实现 综合计算器
栈的基本概念、应用场景以及如何使用数组和单链表模拟栈,并展示了如何利用栈和中缀表达式实现一个综合计算器。
16 1
数据结构与算法学习七:栈、数组模拟栈、单链表模拟栈、栈应用实例 实现 综合计算器
|
11天前
初步认识栈和队列
初步认识栈和队列
36 10
|
5天前
数据结构(栈与列队)
数据结构(栈与列队)
11 1
|
11天前
|
算法
数据结构与算法二:栈、前缀、中缀、后缀表达式、中缀表达式转换为后缀表达式
这篇文章讲解了栈的基本概念及其应用,并详细介绍了中缀表达式转换为后缀表达式的算法和实现步骤。
28 3