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

本文涉及的产品
云数据库 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
目录
相关文章
|
4天前
|
存储 缓存 NoSQL
深入浅出Redis(一):对象与数据结构
深入浅出Redis(一):对象与数据结构
|
2天前
|
存储 NoSQL Redis
Redis数据结构精讲:选择与应用实战指南
Redis数据结构精讲:选择与应用实战指南
13 0
|
4天前
|
存储 NoSQL 算法
深入浅出Redis(十一):Redis四种高级数据结构:Geosptial、Hypeloglog、Bitmap、Bloom Filter布隆过滤器
深入浅出Redis(十一):Redis四种高级数据结构:Geosptial、Hypeloglog、Bitmap、Bloom Filter布隆过滤器
|
4天前
|
C++
数据结构(顺序表 动态定义
数据结构(顺序表 动态定义
11 2
|
13天前
|
存储 NoSQL 关系型数据库
redis数据结构与应用场景
Redis 是一款开源、免费的内存数据库,常用于处理高并发和大数据场景下的热点数据访问,以提升性能。它支持 key-value 存储及多种数据结构,如字符串、列表、集合和哈希表。数据可持久化到磁盘,与 MySQL 等传统数据库相比,Redis 作为缓存能提供更快的读写速度。Redis 应用场景包括:使用字符串进行计数(如商品库存、点赞数)、利用列表实现消息队列或展示最新商品、使用集合去重和计算交集等,以及通过有序集合进行自动排序(如商品热度榜)。
|
18天前
|
存储 NoSQL 算法
Redis入门到通关之Redis数据结构-Hash篇
Redis入门到通关之Redis数据结构-Hash篇
22 1
|
18天前
|
存储 NoSQL Redis
Redis入门到通关之Redis数据结构-Set篇
Redis入门到通关之Redis数据结构-Set篇
20 1
|
6天前
|
机器学习/深度学习 算法 测试技术
【单调栈】3113. 边界元素是最大值的子数组数目
【单调栈】3113. 边界元素是最大值的子数组数目
|
4天前
栈的基本应用
栈的基本应用
10 3
|
4天前
栈与队列理解
栈与队列理解
10 1