2.2 Redis中SDS(简单动态字符串) 与C字符串的区别

news2024/11/29 10:45:17

引言:

根据传统,C语言使用长度N+1的字符数组来表示长度为N的字符串,并且字符数组的最后一个元素总是空字符’\0’。
例如,图2-3 就展示了一个值为"Redis"的C字符串。
C语言使用这种简单的字符串表示方式,并不能满足Redis对字符串在安全性、效率以及功能方面的要求,接下来的内容将详细对比C字符串和SDS之间的区别,并说明SDS比C字符串更适合用于Redis的原因。

在这里插入图片描述

2.2.1 Redis 可以,以常数复杂度 获取 字符串的长度

在传统的C字符串中,其本身,并不记录字符串的长度信息,如果想获取该字符串的长度,那么就需要对字符串进行遍历,对遇到的每个字符进行计数,知道遇到 0元素或者 字符’\0’,这个操作的复杂度为O(N)。

和C字符串不同的是,Redis 中 SDS 在len 属性中记录了SDS本身的长度,所以获取一个SDS长度的复杂度仅为O(1)。

设置和更新SDS长度的工作,是由SDS的API 在执行时自动完成的,所以使用SDS无需进行任何手动修改长度工作。

通过使用SDS这种字符串结构,Redis 把 获取一个字符串的效率,从O(N) 时间复杂度降低到O(1),即使我们对一个非常长的字符串键,进行反复的执行STRLEN 命令,也不会对系统性能造成任何影响,因为STRLEN命令的时间复杂度仅为O(1)。

2.2.2 杜绝了缓冲区溢出

传统C字符串,除了获取字符串长度以外,还会造成缓冲区溢出的问题。

例如如果 对一个已经分配好内存空间的C语言字符串 char *dest,如果这个时候要对该字符串进行追加的话,那么如果在没有追加之前没有为dest额外分配额外的空间(这往往是粗心的程序猿所做的,但是C语言中确实是这样的。每一次对字符串进行新增或者修改,都要进行一次内存分配),一旦没有提前进行分配,那么就会产生缓冲溢出。

举个例子,加入程序中,有2个在内存中紧邻的C字符串s1和s2,其中s1保存了"redis",
s2保存了 “mysql5.7”,如图2-7所示。

在这里插入图片描述

如果一个程序员,决定通过执行:

strcat(s1," Java");

那么s1的内容将会被修改为"redis JAVA"(10个字符),但是粗心的他却忘了在执行此函数之前,为s1分配足够的内存空间,那么在strcat函数执行之后,s1的数据将会溢出到s2所在的空间中,会导致s2保存的内容以外地被修改,如果2-8所示。

在这里插入图片描述

与C字符串不同的是,SDS 的空间分配策略完全杜绝了发生缓冲区溢出的可能性:当SDS的API需要对SDS进行修改时,API 会先检查SDS 的空间是否满足修改所需的要求,如果不满足的话,API 会自动将SDS 的空间扩展至执行修改所需的大小,然后才执行实际的修改操作问题,所以使用SDS既然不需要手动修改SDS的空间大小,也不会出现前面所说的缓冲区溢出问题。

举个例子,SDS的API里面也有一个用于执行拼接操作的sdscat函数,可以将一个C字符串拼接到给定SDS所保存的字符串的后面,但是在执行拼接操作之前,sdscat会先检查给定SDS的空间是否足够,如果不够的话,sdscat就会先扩展SDS的空间,然后才执行拼接操作。

例如,如果我们执行:

sdscat(s," Cluster");

在这里插入图片描述

其中SDS值s 如图2-9 所示,那么sdscat将在执行拼接操作之前检查s的长度是否足够,在发现s目前的空间不足以拼接" Cluster"之后,sdscat就会扩展s的空间,然后才执行拼接
" Cluster"的操作,拼接操作完成之后的SDS如图2-10所示。
在这里插入图片描述
注意:图2-10所示的SDS,sdscat不仅对这个SDS进行了拼接操作,它还为SDS分配了13字节的未使用空间,并且拼接之后的字符串也正好是13字节长,这种现象既不是bug也不是巧合,它和SDS的空间分配策略有关。

链接: 2.2.3 Redis中的SDS减少修改字符串时带来的内存重分配次数

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/49186.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

计算机组成原理习题课第四章-3(唐朔飞)

计算机组成原理习题课第四章-3(唐朔飞) ✨欢迎关注🖱点赞🎀收藏⭐留言✒ 🔮本文由京与旧铺原创,csdn首发! 😘系列专栏:java学习 💻首发时间:&…

Android面试题——高级开发面试题一

一 面试题概述 请简单的分析一下Android系统启动流程的原理?App启动状态有哪几种,各自的启动流程是怎么样的?当项目中遇到黑白屏问题,你有什么好的解决方案?如何查看方法内的耗时时间与方法分析?介绍一下A…

英国公派访问学者带家属签证经验分享

英国公派访问学者带家属签证经验分享,下面就随知识人网老师一起来看一看。 一、学历学位证书 英国签证中心要求提供,但留服网上似乎没有提及。 要是带家属,家属属于Academic Dependants签证。首先介绍一个总的说法,也是据网友提…

游戏测试是一个怎样的行业?

游戏测试真的是玩游戏吗? 游戏测试和软件测试又有什么区别呢? 游戏测试是不是没有前景?能从事吗? 很多人都关注这个问题,所以接下来我详细给大家来介绍下游戏测试。 为了让大家更有获得感,所以本文的行…

【关于我接触了Uview的Upload】——单图上传,多图上传,遇到的问题总结、直传阿里云Oss

Uview的Upload组件 前言 有很长一段时间没有更新了,由于工作的繁忙导致没有时间写博客,今天在做到公司特殊场景需要用到上传组件并直传阿里云Oss,这里简单讲讲我在完成前端项目中使用到Uview的Upload遇到的问题以及我是如何解决&#xff0c…

百万企业用户选择的华为云云服务器,你不来了解一下吗?

今天,企业对“云”的需求像水、电、天然气一样普遍,华为云服务器作为新产品迅速占领现有主机市场。但市面上云服务器的种类多不胜数,怎样才能选到适合自己的云服务器呢?在这里,我推荐华为云云服务器,因为它…

力扣hot100——第2天:4寻找两个正序数组的中位数、5最长回文子串、10正则表达式匹配

文章目录1.4寻找两个正序数组的中位数1.1.题目1.2.解答1.2.1.直接法:合并数组再求结果1.2.2.分治:无需合并数组1.2.3.log(n)的解法2.5最长回文子串3.10正则表达式匹配3.1.题目3.2.解答1.4寻找两个正序数组的中位数 参考:力扣题目链接&#x…

压力串级控制装置用于气动马达的高精度调节

摘要:气动马达作为一种将压缩空气的压力能转换为旋转机械能的装置,其运行的关键是要进行驱动气体压力的控制。本文介绍了目前气动马达压力控制装置的技术现状,特别指出了现有技术中使用电空变换器存在的不足,介绍了电空变换器的更…

深入理解java虚拟机:类加载及执行子系统的案例

文章目录1. 概述2. Tomcat:正统的类加载器结构3. OSGi:灵活的类加载器架构4. 字节码生成技术与动态代理的实现5. Retrotranslator:跨越JDK版本1. 概述 在Class文件格式与执行引擎这部分里,用户的程序能直接影响的内容并不太多&am…

Tableau数据分析数据可视化分析平台

Tableau数据分析&数据可视化分析平台​ 本文章内涉及的资源包以及素材均来自于互联网,仅供大家用来交流学习与研究使用,努力提升自己的一篇文章。各类安装包以及素材版权归属原版权方所有,版权争议与本人无关,用户本人下载后不…

Linux用户操作

用户操作 1、切换用户 root登录 切换到普通用户 howard 命令:su howard 再切换到root用户 一个是执行exit命令,二个是执行su root 查看环境变量 命令:env 切换到Howard用户查看 2、查看用户 查看所有用户 命令:cat …

深入理解Linux网络技术内幕(十)——帧的接收

文章目录前言与其他功能交互设备的开启和关闭队列通知内核帧已接收:NAPI和netif_rxNAPI简介NAPI所用之net_device字段net_rx_action和NAPI新旧驱动程序接口操作poll_list设备驱动程序与内核间的旧接口:netif_rx的第一部分netif_rx的初始任务管理队列以及…

2022最新鸽哒IM即时通讯系统源码 带安卓、苹果、PC端(全开源)+部署教程

提示:即时通讯,纯原生开发,各种功能应有尽有 内容目录一、详细介绍二、效果展示1.部分代码2.效果图展示三、学习资料下载一、详细介绍 提示:即时通讯,纯原生开发,各种功能应有尽有 鸽哒是一款类似于v的即时…

供应生物素PEG衍生物试剂Biotin-PEG-NHS,NHS-PEG-Biotin,生物素PEG活性酯

1、名称 英文:Biotin-PEG-NHS,NHS-PEG-Biotin 中文:生物素-聚乙二醇-活性酯 2、CAS编号:N/A 3、所属分类:Biotin PEG NHS ester PEG 4、分子量:可定制,活性酯-聚乙二醇2-生物素&#xff0c…

Java RMI 远程代码执行漏洞

0x01 漏洞描述 - Java RMI 远程代码执行漏洞 - Java RMI服务是远程方法调用,是J2SE的一部分,能够让程序员开发出基于JAVA的分布式应用。一个RMI对象是一个远程Java对象,可以从另一个Java虚拟机上(甚至跨过网络)调用它…

SpringBoot项目打印接口请求日志,CommonsRequestLoggingFilter实现方式

文章目录需求背景效果图实现思路其他方案对比优缺点分析具体实现需求背景 线上项目出现bug时,可以通过接口的请求参数来排查定位问题。和业务方battle时,能够回怼他是自己操作的问题。 效果图 实现思路 Spring提供了CommonsRequestLoggingFilter过滤器…

opengl,opengl es,egl,glfw,glew

OpenGL ES之GLFW窗口搭建 - Plato - 博客园概述 本章节主要总结如何使用GLFW来创建Opengl窗口。主要包括如下内容: OpenGl窗口创建介绍 GLFW Window版编译介绍 GLFW简单工程源码介绍 OpenGL窗口创建介绍 能用于Ohttps://www.cnblogs.com/feng-sc/p/5093262.htmlOp…

事务-P26,P27

事务:要么都成功,要么都失败。例子:支付宝交易。 acid原则。 11直接移植10的代码 spring-11-transaction: UserMapper(增加两个接口) package com.Li.mapper;import com.Li.pojo.User;import java.util.…

C# 消息 界面卡顿 界面进程 工作进程

一 消息 消息与消息循环,是所有的GUI开发里共同的概念: 消息Message,有的地方也叫事件; ① 鼠标消息; ② 键盘消息; ③ 绘制事件; ④ 窗口最大化、最小化; 1 消息循环 消息循环,M…

10道不得不会的缓存面试题【缓存】

博主介绍: 🚀自媒体 JavaPub 独立维护人,全网粉丝15w,csdn博客专家、java领域优质创作者,51ctoTOP10博主,知乎/掘金/华为云/阿里云/InfoQ等平台优质作者、专注于 Java、Go 技术领域和副业。🚀 最…