理解Redis SDS(简单动态字符串)

2024-01-19 22:38

本文主要是介绍理解Redis SDS(简单动态字符串),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

简述

Redis中并没有直接使用C语言的字符串表示(在内存中以“\0”作为字符串的结尾),而是自己新建了一个字符串的结构,该结构就是SDS(Simple Dynamic String,简单动态字符串)。

SDS在Redis中的使用

Redis中设计可变字符串都是用的SDS结构,如Redis底层为键值对结构,所有的键都是字符串类型的,就是SDS结构的对象。当然Redis打印日志时,字符串不需要变化,则是使用的C语言的字符串。

SDS结构

struct sdshdr{//buf数组中未使用的字节数量int free;//buf数组中已经使用的字节数量,不包括“\0”int len;//字节数组,用来保存字符串char buf[];
}

假如用SDS来存储字符串“Redis”,那么其结构如下:
在这里插入图片描述
len为5,实际buf数组的长度为6,有一个字节用来存储了“\0”。SDS不需要“\0”来判断字符串是否到结尾,那么为什么还要在buf数组最后用“\0”作为结尾呢?其实主要是Redis还要用到C语言中对字符串的处理函数,以“\0”作为结尾,Redis就可以使用这些函数而不用自己去实现。

SDS与C语言字符串的区别

获取字符串长度时时间复杂度O(1)

在C语言的字符串中,以"\0"作为结尾,在需要求字符串长度的时候,需要遍历数组到“\0”处,时间复杂度为O(N)。而在SDS的结构中,可以直接从len中读取长度,时间复杂度为O(1)。

缓冲区溢出问题

传统C语言字符串在执行诸如字符串相加的函数时,是假设使用者已经进行了内存的分配,有足够的内存可以供操作,假如使用者在执行函数之前没有进行内存处理,就可能出现缓冲区溢出的情况。
如:现在有两个字符串S1,S2,结构如下:
在这里插入图片描述
此时,如果直接给S1加上“ Cluster”,就会发生溢出,结构变为:
在这里插入图片描述
可以看到S2的字符串变为“Cluster”。而SDS结构中free字段,明确规定了buf数组空闲字节的数量,执行操作之前可以计算free是否足够放下新的字符串,如果不够就进行内存分配,这样就可以防止缓冲区溢出。

减少内存分配的次数

传统的C语言字符串,在进行字符串相加操作之前,每一次操作都需要进行内存分配,防止缓冲区溢出。而SDS,如果free空闲字节数量足够,完全可以直接进行操作,不需要进行空间申请。那么SDS是怎么判断应该预留多少free的呢?

空间预分配

假如free中空闲的字节数不足以放下新的字符串,那么SDS也会申请进行内存分配,保证buf足够的空间放下新的字符串。当存储的字符串大小小于1MB时,free会和len一样大,也就是假设新的字符串大小len为13,那么free也为13,buf数组的长度为13 + 13 + 1(\0占一位)。如果新的字符串超过1MB,那么free就为1MB。

惰性释放内存空间

buf存储的字符串进行缩减操作时,内存空间并不会立刻释放,释放出来的字节数目会加到free中。

二进制安全

传统的C语音字符串是无法保存媒体数据的,因为如果数据中有“\0”就会被判断为结尾。使用SDS就可以存储媒体数据。

注:"\0"是 ASCII 码为 0 的字符,对应的字符是(Null),C语言中用来判断字符串的结尾

这篇关于理解Redis SDS(简单动态字符串)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/623924

相关文章

Python实现字典转字符串的五种方法

《Python实现字典转字符串的五种方法》本文介绍了在Python中如何将字典数据结构转换为字符串格式的多种方法,首先可以通过内置的str()函数进行简单转换;其次利用ison.dumps()函数能够... 目录1、使用json模块的dumps方法:2、使用str方法:3、使用循环和字符串拼接:4、使用字符

Redis 基本数据类型和使用详解

《Redis基本数据类型和使用详解》String是Redis最基本的数据类型,一个键对应一个值,它的功能十分强大,可以存储字符串、整数、浮点数等多种数据格式,本文给大家介绍Redis基本数据类型和... 目录一、Redis 入门介绍二、Redis 的五大基本数据类型2.1 String 类型2.2 Hash

深入理解Mysql OnlineDDL的算法

《深入理解MysqlOnlineDDL的算法》本文主要介绍了讲解MysqlOnlineDDL的算法,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小... 目录一、Online DDL 是什么?二、Online DDL 的三种主要算法2.1COPY(复制法)

Redis中Hash从使用过程到原理说明

《Redis中Hash从使用过程到原理说明》RedisHash结构用于存储字段-值对,适合对象数据,支持HSET、HGET等命令,采用ziplist或hashtable编码,通过渐进式rehash优化... 目录一、开篇:Hash就像超市的货架二、Hash的基本使用1. 常用命令示例2. Java操作示例三

Redis中Set结构使用过程与原理说明

《Redis中Set结构使用过程与原理说明》本文解析了RedisSet数据结构,涵盖其基本操作(如添加、查找)、集合运算(交并差)、底层实现(intset与hashtable自动切换机制)、典型应用场... 目录开篇:从购物车到Redis Set一、Redis Set的基本操作1.1 编程常用命令1.2 集

Redis中的有序集合zset从使用到原理分析

《Redis中的有序集合zset从使用到原理分析》Redis有序集合(zset)是字符串与分值的有序映射,通过跳跃表和哈希表结合实现高效有序性管理,适用于排行榜、延迟队列等场景,其时间复杂度低,内存占... 目录开篇:排行榜背后的秘密一、zset的基本使用1.1 常用命令1.2 Java客户端示例二、zse

Redis中的AOF原理及分析

《Redis中的AOF原理及分析》Redis的AOF通过记录所有写操作命令实现持久化,支持always/everysec/no三种同步策略,重写机制优化文件体积,与RDB结合可平衡数据安全与恢复效率... 目录开篇:从日记本到AOF一、AOF的基本执行流程1. 命令执行与记录2. AOF重写机制二、AOF的

Spring Gateway动态路由实现方案

《SpringGateway动态路由实现方案》本文主要介绍了SpringGateway动态路由实现方案,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随... 目录前沿何为路由RouteDefinitionRouteLocator工作流程动态路由实现尾巴前沿S

Python 常用数据类型详解之字符串、列表、字典操作方法

《Python常用数据类型详解之字符串、列表、字典操作方法》在Python中,字符串、列表和字典是最常用的数据类型,它们在数据处理、程序设计和算法实现中扮演着重要角色,接下来通过本文给大家介绍这三种... 目录一、字符串(String)(一)创建字符串(二)字符串操作1. 字符串连接2. 字符串重复3. 字

Python动态处理文件编码的完整指南

《Python动态处理文件编码的完整指南》在Python文件处理的高级应用中,我们经常会遇到需要动态处理文件编码的场景,本文将深入探讨Python中动态处理文件编码的技术,有需要的小伙伴可以了解下... 目录引言一、理解python的文件编码体系1.1 Python的IO层次结构1.2 编码问题的常见场景二