欢迎来到尧图网

客户服务 关于我们

您的位置:首页 > 财经 > 金融 > Redis 源码硬核解析系列专题 - 第二篇:核心数据结构之SDS(Simple Dynamic String)

Redis 源码硬核解析系列专题 - 第二篇:核心数据结构之SDS(Simple Dynamic String)

2025/4/2 9:30:29 来源:https://blog.csdn.net/yinlongfei_love/article/details/146705102  浏览:    关键词:Redis 源码硬核解析系列专题 - 第二篇:核心数据结构之SDS(Simple Dynamic String)
1. 引言

Redis没有直接使用C语言的标准字符串(以\0结尾的字符数组),而是自定义了SDS(Simple Dynamic String)。SDS是Redis的基础数据结构之一,广泛用于键值存储、命令参数等场景。本篇将深入剖析SDS的实现原理、优势以及源码细节。


2. 为什么不用C标准字符串?

C字符串存在以下问题:

  • 缓冲区溢出strcat等操作可能越界。
  • 长度计算:需要遍历到\0,时间复杂度O(n)。
  • 内存管理:频繁的拼接和释放效率低下。

SDS通过额外的元数据和优化策略,解决了这些问题,成为Redis高性能的基石。


3. SDS的结构体定义

SDS的定义在src/sds.hsrc/sds.c中。Redis 3.2之后引入了多种SDS类型以节省内存,但核心思想一致。我们以最基本的SDS结构为例:

代码片段sds.h):

typedef char *sds;struct sdshdr {unsigned int len;    // 已使用长度unsigned int free;   // 未使用长度char buf[];          // 实际存储数据的柔性数组
};
  • len:记录字符串的实际长度,避免遍历。
  • free:记录剩余可用空间,支持动态扩展。
  • buf:存储字符串内容,紧跟结构体。

硬核点:SDS的内存布局是连续的,sds指针直接指向buf,而通过指针偏移可以访问sdshdr

Mermaid内存布局图

classDiagramclass SDS {-len: uint-free: uint-buf: char[]}note for SDS "sds指针指向buf起始地址"

4. SDS的核心操作解析
4.1 创建SDS(sdsnew()

代码片段sds.c):

sds sdsnew(const char *init) {size_t initlen = (init == NULL) ? 0 : strlen(init);return sdsnewlen(init, initlen);
}sds sdsnewlen(const void *init, size_t initlen) {struct sdshdr *sh;size_t alloc = initlen;sh = zmalloc(sizeof(struct sdshdr) + alloc + 1); // +1 为\0sh->len = initlen;sh->free = 0;if (init) memcpy(sh->buf, init, initlen);sh->buf[initlen] = '\0';return (char*)sh->buf;
}

硬核解析

  • zmalloc():Redis自定义的内存分配器。
  • 内存分配包括sdshdr头部和buf(加1字节存放\0以兼容C函数)。
  • 返回值是buf的地址,隐藏了头部信息。
4.2 拼接SDS(sdscat()

代码片段sds.c):

sds sdscat(sds s, const char *t) {return sdscatlen(s, t, strlen(t));
}sds sdscatlen

版权声明:

本网仅为发布的内容提供存储空间,不对发表、转载的内容提供任何形式的保证。凡本网注明“来源:XXX网络”的作品,均转载自其它媒体,著作权归作者所有,商业转载请联系作者获得授权,非商业转载请注明出处。

我们尊重并感谢每一位作者,均已注明文章来源和作者。如因作品内容、版权或其它问题,请及时与我们联系,联系邮箱:809451989@qq.com,投稿邮箱:809451989@qq.com

热搜词