【Redis】数据结构---String-创新互联

文章目录
    • String(字符串)
      • 1.Redis 键(key)
      • 2.String(字符串)
        • 2.1常用命令
        • 2.2.String底层结构
      • 3.空间分配策略
        • 3.1空间预分配
        • 3.2惰性空间释放
        • 3.3为什么SDS的大长度是512M?
      • 4.SDS面试题

成都创新互联-专业网站定制、快速模板网站建设、高性价比商洛网站开发、企业建站全套包干低至880元,成熟完善的模板库,直接使用。一站式商洛网站制作公司更省心,省钱,快速模板网站建设找我们,业务覆盖商洛地区。费用合理售后完善,10余年实体公司更值得信赖。String(字符串) 1.Redis 键(key)
  • keys *查看当前库所有key (匹配:keys *1)
  • exists key判断某个key是否存在
  • type key 查看你的key是什么类型
  • del key 删除指定的key数据
  • unlink key 根据value选择非阻塞删除。仅将keys从keyspace元数据中删除,真正的删除会在后续异步操作。
  • expire key 10 10秒钟:为给定的key设置过期时间
  • ttl key 查看还有多少秒过期,-1表示永不过期,-2表示已过期

image-20221130014337304

执行上面的操作

image-20221130015234931

2.String(字符串)

String是Redis最基本的类型,一个key对应一个value。

String类型是二进制安全的。意味着Redis的string可以包含任何数据。比如jpg图片或者序列化的对象。

String类型是Redis最基本的数据类型,一个Redis中字符串value最多可以是512M

2.1常用命令

set 添加键值对

image-20221130021209021

关于set的其他参数

  • NX:当数据库中key不存在时,可以将key-value添加数据库
  • *XX:当数据库中key存在时,可以将key-value添加数据库,与NX参数互斥
  • EX:key的超时秒数
  • PX:key的超时毫秒数,与EX互斥

get 查询对应键值

append 将给定的 追加到原值的末尾

strlen 获得值的长度

setnx 只有在 key 不存在时 设置 key 的值

image-20221130022154240

数字类型的命令:

incr

  • 将 key 中储存的数字值增1
  • 只能对数字值操作,如果为空,新增值为1

decr

  • 将 key 中储存的数字值减1
  • 只能对数字值操作,如果为空,新增值为-1

incrby / decrby<步长>将 key 中储存的数字值增减。自定义步长。

  • 注意:INCR类型的命令是原子的,Redis单命令的原子性主要得益于Redis的单线程。

image-20221130022352456

mset …

  • 同时设置一个或多个 key-value对

mget …

  • 同时获取一个或多个 value

image-20221130022635543

msetnx …

  • 同时设置一个或多个 key-value 对,当且仅当所有给定 key 都不存在。
  • 该命令具有原子性:其中有一个key是存在的,则都创建失败

image-20221130022855031

getrange<起始位置><结束位置>

  • 获得值的范围,类似C++中string类的substr。

setrange<起始位置>

  • 用 覆写所储存的字符串值,从<起始位置>开始(索引从0****开始)。

image-20221130023035175

setex<过期时间>

  • 设置键值的同时,设置过期时间,单位秒。

getset

  • 以新换旧,设置了新值同时获得旧值。
2.2.String底层结构

Redis是用C语言写的,但是对应Redis的Sting,并不是C 语言中的字符串(即以空字符’\0’结尾的字符数组);Redis自定义了数据结构SDS(simple dynamic string)【简单动态字符串】,并将 SDS 作为 Redis的默认字符串表示。

struct sdshdr{//记录 buf 数组中未使用字节的数量
     int free;
    
    //记录buf数组已使用字节的数量
    //等于 SDS 保存字符串的长度
     int len;
        
     //字节数组,用于保存字符串
     char buf[];	//柔性数组
}

image-20221130024613783

  • len为字符串的实际长度,保证了获取字符串长度时为O(1)操作
  • free为buf数组中剩余的空间大小
  • buf 保存字符串的数组
  • 如果长度不够,会主动申请空间
  • redis实际开辟的空间为len+free

优点:

减少修改字符串的内存重新分配次数

C语言字符串由于不记录字符串长度,所以如果要修改字符串,必须重新分配内存(先释放再申请)。因为如果没有重新分配,字符串长度增大时会造成内存缓冲区溢出,字符串长度减小时会造成内存泄露。

对于SDS,由于len属性和free属性的存在,对于修改字符串SDS实现了空间预分配和惰性空间释放两种策略:

  • **空间预分配:**对字符串进行空间扩展的时候,扩展的内存比实际需要的多,这样可以减少连续执行字符串增长操作所需的内存重分配次数。
  • **惰性空间释放:**对字符串进行缩短操作时,程序不立即使用内存重新分配来回收缩短后多余的字节,而是使用 free 属性将这些字节的数量记录下来,等待后续使用。

二进制安全

  • C字符串以空字符作为字符串结束的标识,而对于一些二进制文件(如图片等),内容可能包括空字符串,因此C字符串无法正确存取;
  • 而所有 SDS 的API 都是以处理二进制的方式来处理 buf 里面的元素,**并且 SDS 不是以空字符串来判断是否结束,而是以 len 属性表示的长度来判断字符串是否结束。**所以SDS可以保存图片、音频等文件内容。
3.空间分配策略 3.1空间预分配

C++中数组在进行扩容时,往往会申请一个更大的数组,然后把数组拷贝过去。Redis同样基于这种策略提高了空间预分配机制。

当执行字符串增长操作并且需要扩展内存时,程序不仅仅会给SDS分配必需的空间还会分配额外的未使用空间,其长度存到free属性中。具体如下:

  • **如果修改后len长度将小于1M,这时分配给free的大小和len一样;**例如修改过后为10字节, 那么给free也是10字节,buf实际长度变成了10+10+1 = 21byte(别忘记了\0的存在)
  • **如果修改后len的长度大于等于1M,这时分配给free的长度为1M;**例如修改过后为30M,那么给free是1M.buf实际长度变成了30M+1M+1byte

image-20221130031707362

3.2惰性空间释放

惰性空间释放用于字符串缩短的操作。当字符串缩短是,程序并不是立即使用内存重分配来回收缩短出来的字节,而是使用free属性记录起来,并等待将来使用。

image-20221130031932892

3.3为什么SDS的大长度是512M?

Redis字符串使用int类型表示长度,一共有32个比特位。2^32字节=512M

4.SDS面试题

1.SDS如何兼容C语言字符串?如何保证二进制安全?

C99中提到,结构体的最后一个成员如果是一个数组,大小不确定,那么就是一个柔性数组。

SDS对象中的buf是一个柔性数组,上层调用时,SDS直接返回了buf。由于buf是直接指向内容的指针,所以兼容C语言函数。而当真正读取内容时,SDS会通过len来限制读取长度,而非“\0”,所以保证了二进制安全。

2.SDS是如何扩容的?

空间预分配。先判断扩容长度与free的大小关系,如果够就直接拼接字符串,如果不够使用空间预分配的方式扩容。

你是否还在寻找稳定的海外服务器提供商?创新互联www.cdcxhl.cn海外机房具备T级流量清洗系统配攻击溯源,准确流量调度确保服务器高可用性,企业级服务器适合批量采购,新人活动首月15元起,快前往官网查看详情吧


网站名称:【Redis】数据结构---String-创新互联
网站链接:http://pcwzsj.com/article/piocg.html