Redis入门到通关之数据结构解析-动态字符串SDS

本文涉及的产品
云数据库 Redis 版,标准版 2GB
推荐场景:
搭建游戏排行榜
云原生内存数据库 Tair,内存型 2GB
全局流量管理 GTM,标准版 1个月
简介: Redis入门到通关之数据结构解析-动态字符串SDS

Redis数据结构-动态字符串


我们都知道 Redis 中保存的Key是字符串,value 往往是字符串或者字符串的集合。可见字符串是 Redis 中最常用的一种数据结构。

不过 Redis 没有直接使用C语言中的字符串,因为C语言字符串存在很多问题:

  • 获取字符串长度的需要通过运算
  • 非二进制安全
  • 不可修改

Redis 构建了一种新的字符串结构,称为简单动态字符串(Simple Dynamic String),简称 SDS

例如,我们执行命令:

set name 技术

那么 Redis 将在底层创建两个 SDS,其中一个是包含“name”的 SDS,另一个是包含“技术”的 SDS。

Redis 是 C 语言实现的,其中 SDS 是一个结构体,源码如下:

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
typedef struct sds {
    int len;         // 记录字符串的长度
    int free;        // 记录未使用的字节数
    char buf[];      // 字符串的实际存储空间
} sds;
// 创建一个新的 sds
sds *sds_new(int init_len) {
    sds *s = malloc(sizeof(sds) + init_len + 1);
    if (!s) return NULL;
    s->len = 0;
    s->free = init_len;
    s->buf[0] = '\0';
    return s;
}
// 释放 sds
void sds_free(sds *s) {
    if (s) free(s);
}
// 追加字符串到 sds
sds *sds_append(sds *s, const char *append) {
    int append_len = strlen(append);
    if (s->free < append_len) {
        s = realloc(s, sizeof(sds) + s->len + append_len + 1);
        if (!s) return NULL;
        s->free = append_len;
    }
    memcpy(s->buf + s->len, append, append_len);
    s->len += append_len;
    s->free -= append_len;
    s->buf[s->len] = '\0';
    return s;
}
// 打印 sds
void sds_print(const sds *s) {
    printf("Length: %d\n", s->len);
    printf("Free: %d\n", s->free);
    printf("String: %s\n", s->buf);
}
int main() {
    sds *str = sds_new(10);
    if (!str) {
        printf("Error: Unable to create SDS.\n");
        return 1;
    }
    str = sds_append(str, "Hello, ");
    if (!str) {
        printf("Error: Unable to append to SDS.\n");
        return 1;
    }
    str = sds_append(str, "world!");
    if (!str) {
        printf("Error: Unable to append to SDS.\n");
        return 1;
    }
    sds_print(str);
    sds_free(str);
    return 0;
}

例如,一个包含字符串“name”的sds结构如下:


动态扩容举例


SDS 之所以叫做动态字符串,是因为它具备动态扩容的能力,例如一个内容为“hi”的SDS:

假如我们要给SDS追加一段字符串“,Amy”,这里首先会申请新内存空间:

  • 如果新字符串小于1M,则新空间为扩展后字符串长度的两倍+1;
  • 如果新字符串大于1M,则新空间为扩展后字符串长度+1M+1。称为内存预分配


二进制安全


  • Redis 可以存储各种数据类型,那么二进制数据肯定也不例外。但二进制数据并不是规则的字符串格式,可能会包含一些特殊的字符,比如 ‘\0’ 等。
  • 前面我们提到过,C 中字符串遇到 ‘\0’ 会结束,那 ‘\0’ 之后的数据就读取不上了。但在 SDS 中,是根据 len 长度来判断字符串结束的。
  • 如此, 二进制安全的问题就解决了。


SDS优点


SDS 优点:

  1. 获取字符串长度的时间复杂度为O(1)
  2. 支持动态扩容, 减少内存分配次数
  3. 二进制安全

与C语言中的字符串的区别


SDS(Simple Dynamic String)是Redis使用的一种字符串表示方式,与C语言中的字符串有一些显著的区别:

  • 动态调整大小:SDS是动态分配内存的,可以根据需要动态增加或减少其大小,而C语言中的字符串通常是静态分配的,大小在创建时就确定了,后续无法直接调整大小。
  • 二进制安全:SDS允许存储任意二进制数据,而C语言中的字符串通常以NULL结尾,因此不适合存储包含NULL字符的二进制数据。
  • 惰性空间释放:SDS采用惰性空间释放策略,即当SDS的内容被截断或缩短时,并不立即释放多余的内存,而是等待下一次需要扩展空间时才释放。这种策略可以减少频繁的内存分配和释放操作,提高性能。而C语言中的字符串在缩短时无法自动释放多余的内存,需要手动管理内存。
  • 长度存储:SDS在内部存储了字符串的长度信息,这样可以在O(1)时间内获取字符串的长度,而C语言中的字符串需要遍历整个字符串才能确定长度。

SDS是一种更加灵活和高效的字符串表示方式,特别适合需要频繁修改和操作字符串的场景,而C语言中的字符串更适合于静态不变的情况。

相关实践学习
基于Redis实现在线游戏积分排行榜
本场景将介绍如何基于Redis数据库实现在线游戏中的游戏玩家积分排行榜功能。
云数据库 Redis 版使用教程
云数据库Redis版是兼容Redis协议标准的、提供持久化的内存数据库服务,基于高可靠双机热备架构及可无缝扩展的集群架构,满足高读写性能场景及容量需弹性变配的业务需求。 产品详情:https://www.aliyun.com/product/kvstore &nbsp; &nbsp; ------------------------------------------------------------------------- 阿里云数据库体验:数据库上云实战 开发者云会免费提供一台带自建MySQL的源数据库&nbsp;ECS 实例和一台目标数据库&nbsp;RDS实例。跟着指引,您可以一步步实现将ECS自建数据库迁移到目标数据库RDS。 点击下方链接,领取免费ECS&amp;RDS资源,30分钟完成数据库上云实战!https://developer.aliyun.com/adc/scenario/51eefbd1894e42f6bb9acacadd3f9121?spm=a2c6h.13788135.J_3257954370.9.4ba85f24utseFl
相关文章
|
4天前
|
消息中间件 NoSQL Redis
redis数据结构-List
redis数据结构-List
16 1
|
4天前
|
存储 缓存 NoSQL
redis数据结构-字符串
redis数据结构-字符串
13 1
|
3天前
|
存储 监控 NoSQL
redis数据结构-HyperLogLog
redis数据结构-HyperLogLog
11 1
|
4天前
|
存储 NoSQL Redis
redis数据结构-ziplist
redis数据结构-ziplist
7 2
|
19天前
|
应用服务中间件 nginx C语言
Nginx入门 -- 基本数据结构中之ngx_str_t,ngx_array_t
这两种数据结构是Nginx自定义数据类型的例子,它们证明了Nginx设计者在构建一个为高并发和高性能优化的web服务器时的精确和高效。理解这些数据结构是深入学习Nginx内部机制的基础,同时也是扩展和开发Nginx模块不可或缺的一部分知识。
23 1
|
3天前
|
存储 NoSQL 数据处理
redis数据结构-Bitmaps
redis数据结构-Bitmaps
8 0
|
3天前
|
存储 缓存 NoSQL
redis数据结构-hash
redis数据结构-hash
5 0
|
15天前
|
存储 缓存 算法
深入解析B树:数据结构、存储结构与算法优势
深入解析B树:数据结构、存储结构与算法优势
|
18天前
|
存储 算法 调度
10种 Python数据结构,从入门到精通
10种 Python数据结构,从入门到精通
21 0
|
2月前
|
算法 数据挖掘 计算机视觉
Python并查集实战宝典:从入门到精通,让你的数据结构技能无懈可击!
【7月更文挑战第17天】并查集,如同瑞士军刀,是解决元素分组问题的利器,应用于好友关系、像素聚类、碰撞检测和连通性分析等场景。本文从基础到实战,介绍并查集的初始化、查找与路径压缩、按秩合并,以及在Kruskal算法中的应用。通过并查集,实现高效动态集合操作,对比哈希表和平衡树,其在合并与查找上的性能尤为突出。学习并查集,提升算法解决复杂问题的能力。
50 5

热门文章

最新文章

推荐镜像

更多
下一篇
云函数