Redis没有直接使用C语言传统的字符串表示(以空字符结尾的字符数组), 而是自己构建了一种名为简单动态字符串(simple dynamic string,SDS)的抽象类型, 并将SDS用作Redis的默认字符串表示。在Redis中,C字符串只会作为字符串字面量,用在一些无需对字符串值进行修改的地方,例如打印日志。

一、SDS的结构定义

示例:

二、SDS与C字符串的区别
1. 常数复杂度获取字符串长度
C字符串长度计算:遍历整个字符串直至遇到代表字符串结尾的空字符,时间复杂度为O(N)。
SDS长度计算:只要访问SDS的'len'属性即可,时间复杂度为O(1)。
注:设置和更新SDS长度的工作是由SDS的API在执行时自动完成的,使用SDS无须进行任何手动修改长度的工作。
2. 杜绝缓冲区溢出
C字符串:增加长度时,需要开发者手动申请分配足够的空间,否则会发生缓冲区溢出。
SDS:SDS的空间分配交由SDS API自动实现,无需手动申请分配,不会发生缓冲区溢出。
3. 减少修改字符串时带来的内存重分配次数
C字符串:每次增长或缩短字符串,都需要进行一次内存重分配操作,否则会发生缓冲区溢出或内存泄漏。
SDS:通过使用"free"属性记录未使用空间,实现空间预分配和惰性空间释放两种优化策略,从而减少内存重分配次数。
(1)SDS空间预分配:
  SDS API对SDS的空间进行扩展的时候,不仅会为SDS分配必需的空间,还会分配额外的未使用空间。若修改后SDS长度小于1M,则预分配和"len"一样大小的未使用空间;若修改后SDS长度大于等于1M,则预分配1M的未使用空间。以后再需要扩展SDS空间时,SDS API会优先使用SDS的"free"未使用空间。
(2)SDS惰性空间释放:
  SDS API对SDS的空间进行缩短的时候,不会立即使用内存重分配来回收多余空间,而是使用"free"属性将这些空间的字节数量记录起来,等待以后使用。SDS也提供了相应的API,让我们可以在有需要时,真正地释放SDS中未使用的空间。
注:内存重分配涉及复杂的算法,并且可能需要执行系统调用,通常是一个比较耗时的操作,若频繁执行可能会对性能造成影响。
4. 二进制安全
C字符串:字符必须符合某种编码,而且除了字符串末尾,不能包含空字符,所以C字符串只能保存文本数据,而不能保存像图片、音频、视频、压缩文件这样的二进制数据。
SDS:SDS API都是二进制安全的,可以保存任意格式的二进制数据。
5. 兼容部分C字符串函数
  SDS API总会将SDS保存的数据末尾设置为空字符,所以若SDS保存的是文本数据,则可以重用一部分<string.h>库定义的函数,避免不必要的代码重复。

三、SDS在Redis中的用途
1. 在Redis的数据库中,包含字符串值的键值对在底层都是由SDS实现的。
2. 除此之外,SDS还被用作缓冲区,例如AOF模块的AOF缓冲区、客户端状态中的输入缓冲区。

Redis数据结构之简单动态字符串的更多相关文章

  1. Redis数据结构之简单动态字符串SDS

    Redis的底层数据结构非常多,其中包括SDS.ZipList.SkipList.LinkedList.HashTable.Intset等.如果你对Redis的理解还只停留在get.set的水平的话, ...

  2. Redis 数据结构之简单动态字符串SDS

    几个概念1:key对象 数据库存储键值对的键,总是一个字符串对象.2:value对象 数据库存储键值对的值,可以是字符串对象,list对象,hash对象,set对象,sorted set对象.    ...

  3. 【redis】redis底层数据结构原理--简单动态字符串 链表 字典 跳跃表 整数集合 压缩列表等

    redis有五种数据类型string.list.hash.set.zset(字符串.哈希.列表.集合.有序集合)并且自实现了简单动态字符串.双端链表.字典.压缩列表.整数集合.跳跃表等数据结构.red ...

  4. redis 系列3 数据结构之简单动态字符串 SDS

    一.  SDS概述 Redis 没有直接使用C语言传统的字符串表示,而是自己构建了一种名为简单动态字符串(simple dynamic string, SDS)的抽象类型,并将SDS用作Redis的默 ...

  5. redis 笔记01 简单动态字符串、链表、字典、跳跃表、整数集合、压缩列表

    文中内容摘自<redis设计与实现> 简单动态字符串 1. Redis只会使用C字符串作为字面量,在大多数情况下,Redis使用SDS(Simple Dynamic String,简单动态 ...

  6. Redis中的简单动态字符串

    Redis没有直接使用C语言传统的字符串表示(以空字符结尾的字符数组,以下简称C字符串),而是自己构建了一种名为简单动态字符串(simple dynamic string,SDS)的抽象类型,并将SD ...

  7. Redis核心原理-简单动态字符串SDS

    SDS简介 Redis是C语言编写的,但没有使用c语言的字符串结构,而是自己实现了一套简单动态字符串 simple dynamic string 简称SDS,SDS兼容C语言的字符串类型,原理类似Ja ...

  8. 关于redis中SDS简单动态字符串

    1.SDS 定义 在C语言中,字符串是以’\0’字符结尾(NULL结束符)的字符数组来存储的,通常表达为字符指针的形式(char *).它不允许字节0出现在字符串中间,因此,它不能用来存储任意的二进制 ...

  9. redis底层数据结构--简单动态字符串 链表 字典 跳跃表 整数集合 压缩列表

    1.动态字符串 redis中使用c语言的字符床存储字面量,默认字符串存储采用自己构建的简单动态字符串SDS(symple dynamic string) redis包含字符串的键值对都是用SDS实现的 ...

随机推荐

  1. python之路——递归函数

    阅读目录 楔子 初识递归 再谈递归 递归函数与三级菜单 递归函数与二分查找算法 楔子 在讲今天的内容之前,我们先来讲一个故事,讲的什么呢?从前有座山,山里有座庙,庙里有个老和尚讲故事,讲的什么呢?从前 ...

  2. POI 读取 Excel 文件

    import java.io.File; import java.io.FileOutputStream; import java.io.InputStream; import java.io.Out ...

  3. Java中的线程--线程的互斥与同步通信

    Java中的线程之前也提到过,但是还是想再详细的学习一下,跟着张孝祥老师,系统的再学习一下. 一.线程中的互斥 线程安全中的问题解释:线程安全问题可以用银行中的转账 例题描述: 线程A与线程B分别访问 ...

  4. shell脚本,如何用shell打印金字塔

  5. non-JRMP server at remote endpoint

    #在相应的domain的domain.xml文件添加下面红色设置,并重启domain <admin-service system-jmx-connector-name="system& ...

  6. 学习C语言库函数

    使用C语言功能强大的函数,我们需要包含头文件 #include<math.h>. 1)取两个数的较大值或较小值函数: double a = 9.9; double b = 6.6; pri ...

  7. sessionStorage 的数据会在同一网站的多个标签页之间共享吗?这取决于标签页如何打开

    一直以来,我所以为的 sessionStorage 的生命周期是这样的:在 sessionStorage 中存储的数据会在当前浏览器的同一网站的多个标签页中共享,并在此网站的最后一个标签页被关闭后清除 ...

  8. [CODEVS] 2189 数字三角形W

    数字三角形 要求走到最后mod 100最大 可达性DP(好像是这样叫) 用bool数组f[i][j][k]表示 位置(i,j)能否得到k(mod 100意义下) 转移条件 f[i][j][k]=f[i ...

  9. MySQL 查询优化之 Index Condition Pushdown

    MySQL 查询优化之 Index Condition Pushdown Index Condition Pushdown限制条件 Index Condition Pushdown工作原理 ICP的开 ...

  10. Mycat主从分离

    1. mycat原理 主从的读写是不同的,主能写能读,再从上写是无法同步到主的,因此需要中间件将主从的读写进行分离,使得主从各司其职,相当于负载均衡的作用.中间件可以是proxy或者mycat.客户端 ...