Bootstrap

Redis(四)value类型——string

目录

一、有关 string 类型的几点注意事项

二、string 类型的常见操作命令

1、set

2、setnx

3、setxx 

4、set setnx setxx的执行流程图

5、mset

6、get

如何解决 value 是中文导致的十六进制字节表示,不易观察的问题?

7、mget

8、多次 get 与 mget 的效率对比

9、append

10、getrange

11、setrange

12、strlen

三、string 类型的常见计数命令

1、incr

2、incrby

3、decr

4、decrby

5、incrbyfloat

四、string 类型命令小结

五、内部编码类型

1、int

2、embstr

3、raw

六、string 类型的几个典型使用场景 

1、作为缓存

2、用来计数

3、共享会话


一、有关 string 类型的几点注意事项

在介绍 string 类型的各种操作之前,我们先介绍几点有关 string 类型的注意事项。

1、Redis 中的所有 key 都是 string 类型的,value 的其它几种数据类型也都是在 string 的基础上构建的,先拥有操作 string 类型的能力,可以为其它几种类型奠定基础

2、Redis 中的字符串都是直接按照二进制数据的形式存储的,不进行任何编码转换;因此相较于 MySQL,其遇到乱码的概率更小。(MySQL 有默认字符集,其默认字符集是拉丁文的,就容易乱码)

3、Redis 中的字符串类型因为是二进制数据形式存储的的,所以不仅仅可以存储文本数据、数字、各种格式的字符串,类似图片、视频、音频等二进制流数据,也都是可以存储的。(不过因为 Redis 要保证速度,一个字符串最大不能超过 512 MB) 

value 的 string 类型,由于其二进制保存的特点,可以存储很多格式的数据

二、string 类型的常见操作命令

1、set

set 命令之前在第一章的时候讲到过,就是设置一个键值对。

如果这个 key 之前存在,其对应的 value 不论是什么类型,都会由新的 value 覆盖,之前的超时时间 TTL 也会失效。

set key value [EX seconds | PX milliseconds] [NX | EX]

我们可以发现,set 命令还支持一些选项来影响其行为:

EX seconds:即以秒级单位设置 key 的过期时间

PX milliseconds:即以毫秒级单位设置 Key 的过期时间

(eg:set key value ex 10  等价于  set key value 10  + expire key 10 两行命令;set key value px 10 等价于 set key value 10 + pexpire 10 两行命令;提高了请求效率)

NX:只在 key 不存在时才进行设置,如果 key 之前已经存在,设置不执行。即新建键值对。

XX:只在 key 存在时才进行设置,如果 key 之前不存在,设置不执行。即相当于更新键值对的 value。 

使用实例

2、setnx

setnx key名 value名

与 set key value nx 一致,是其的简化版命令,即设置键值对,不存在的时候才能设置成功;否则失败。 

3、setxx 

setxx key名 value名

与 set key value xx 一致,是其的简化版命令,即覆盖键值对,存在的时候才能设置成功;否则失败。 

4、set setnx setxx的执行流程图

执行流程符合直觉

5、mset

mset key1 value1 key2 value2……

 mset 顾名思义,就是可以一次性设置多个键值对;与 set 一样,键值对存在即覆盖。(不过并不支持 EX PX NX XX 这些额外选项)

6、get

获取 key 对应的 value,如果 Key 不存在,返回 nil。如果 Key 之前也介绍过。

get key
我们 Key2 是中文,会为十六进制的字节表示

如何解决 value 是中文导致的十六进制字节表示,不易观察的问题?

Redis 客户端在显示中文字符时,默认会将其转义为十六进制的字节表示,以确保不会出现乱码或不可打印字符。因此,当存储一个中文字符串,Redis 会显示其对应的 UTF-8 编码的字节序列。

这时候,我们可以在启动 Redis 客户端的时候加上 --raw 选项,让Redis 客户端以原始模式运行,直接显示未转义的字符串:

可以发现正常显示了

7、mget

mget key1 key2 ……

mget 顾名思义,就是可以一次性获取多个 key 对应的 value 。如果 key 不存在 / value 类型不是 string,返回 nil。 

8、多次 get 与 mget 的效率对比

可以发现,使用 mget 可以有效减少网络交互次数,节省时间,性能更高。 

学会使用批量操作,可以有效提高业务处理效率;但是要注意,每次批量操作所发送的键的数量也不是无节制的,否则可能导致单一命令执行时间过长,造成 Redis 阻塞。

9、append

append key value

append 是追加字符串命令。如果 key 已经存在并且是⼀个 string,append 命令会将 value 追加到原有 string 的后边;如果 key 不存在, 则效果等同于 SET 命令。

返回值是追加后的字符串的字节长度。(因为 Redis 不会对字符做任何处理,所以返回的是字节长度) 

每一个中文字符3个字节

10、getrange

getrange key start开始字节 end结束字节

获取 key 对应的键值对的子串,范围由 start 和 end 确定(注意:区间是左闭右闭)

可以使用负数表示倒数的字节,-1 表示倒数第一个字节,-2 表示倒数第二个字节,以此类推。

(如果字符串保存了汉字,进行子串切分的时候可能会不是完整汉字,需要手动处理,以 3 个字节为单位)

11、setrange

setrange key offset偏移量 value

 setrange 即从第几个字节开始替换字符串的一部分(给出偏移量的地方,多少个字节),结束后返回新的 string 长度。

(中文字符仍然会出现不完整问题,要注意;同时,setrange 也可以针对不存在的键值对操作,然后偏移量之前的字节会被填充为 0x00)

 

12、strlen

strlen key

获取 key 对应的 value 的长度。当 value 存放的类型不是 string 时,报错。(单位是字节)

三、string 类型的常见计数命令

1、incr

incr key名

将 key 对应的 value 表示的数字 + 1;返回 +1 后的数值。

如果 key 不存在,则会创建一个新的键值对,并视 key 对应的 value 大小是 0,然后 + 1。

如果 key 对应的 value 不是⼀个整型或者范围超过了 64 位有符号整型(即 long long),则报错。

2、incrby

incrby key名 要加上多少

将 key 对应的 value 表示的数字加上给的数;返回加法后的数值。(也可以加负数,此时就类似于减法了,但是因为有专门的减法操作,所以不建议这么做)

如果 key 不存在,则会创建一个新的键值对,并视 key 对应的 value 大小是 0,然后加上想要的数。

如果 key 对应的 value 不是⼀个整型或者范围超过了 64 位有符号整型(即 long long),则报错。

3、decr

decr key名

将 key 对应的 value 表示的数字 - 1;返回 -1 后的数值。

如果 key 不存在,则会创建一个新的键值对,并视 key 对应的 value 大小是 0,然后 - 1。

如果 key 对应的 string 不是⼀个整型或者范围超过了 64 位有符号整型,则报错。

4、decrby

decrby key名 要减去多少

将 key 对应的 value 表示的数字减去给的数;返回计算后的数值。(也可以减负数,此时就类似于加法了,但是因为有专门的加法操作,所以不建议这么做)

如果 key 不存在,则会创建一个新的键值对,并视 key 对应的 value 大小是 0,然后减去想要的数。

如果 key 对应的 value 不是⼀个整型或者范围超过了 64 位有符号整型(即 long long),则报错。

5、incrbyfloat

incrbyfloat key 要加的浮点数

将 key 对应的 string 表⽰的浮点数加上对应的值。如果对应的值是负数,则视为减去对应的值。(允许采用科学计数法表示浮点数)

如果 key 不存在,则视为 key 对应的 value 是 0。

如果 key 对应的不是 string,或者不是⼀个浮点数,则报错。

(PS:没有给减法版本,因为 incrbyfloat 用的确实不多,redis 计数主要还是整型计数,因为小数在 Redis 中是按照字符串保存的,进行算术运算,需要把字符串转换成小数,运算完毕后再转换回钱字符串保存,很麻烦)

四、string 类型命令小结

五、内部编码类型

之前在第一章也曾经介绍了,string 类型的内部编码有三种:int, embstr, raw

1、int

int 一般是 8 个字节的长整型。redis 一般也可以用来实现一些计数功能;因此当 value 是整数的时候,redis 底层就会把这个 value 优化成 int 来保存,占用空间小,也便于计算。

2、embstr

压缩字符串,针对短字符串进行特殊优化,一般是字符串小于等于 39 字节的时候,会进行优化。

(39 字节的标准并不一定,有时候根据业务场景进行实时修改;比如某个业务场景的每个键值对的 value 的 string 长度都是 100 字节,这时候也可以考虑都用 embstr )

3、raw

raw 就是最基本的字符串,底层是 char 数组。大于 39 字节的时候,就会使用 raw 类型。

六、string 类型的几个典型使用场景 

1、作为缓存

前文也已经提过,Redis 最主要的作用,就是作为和 MySQL 搭配使用,Redis 作为缓存层,MySQL 作为存储层,然后绝大部分数据都从 Redis 中获取。

由于 Redis 具有支持高并发的特性,所以作为缓存,可以提高加速读写和减少后端压力的作用。

通过增加缓存功能,在理想情况下,每个⽤⼾信息,⼀个⼩时期间只会有⼀次 MySQL 查询,极⼤地提升了查询效率,也降低了 MySQL 的访问数。

2、用来计数

许多应用都会使用 Redis 作为计数的基础供具,它可以实现快速计数、查询缓存的功能,同时数

据可以异步处理或者落地到其他数据源。

如下图所示,例如视频网站的视频播放次数可以使用 Redis 来完成:用户每播放⼀次视频,相应的视频播放数就会自增 1。

写入数据仓库(MySQL、hdfs等)的过程是异步的,不是说来了一个增加播放请求就会立即写入,可以灵活调整

实际中要开发⼀个成熟、稳定的真实计数系统,要面临的挑战其实远没有如此简单:防作弊、按

照不同维度计数、避免单点问题、数据持久化到底层数据源等。

3、共享会话

每个应用服务器,维护着自己的会话数据,如果彼此之间不共享,用户请求一旦因为负载均衡而访问到不同的服务器,就可能出现一些不能正确处理的情况,这时候就可以由 Redis 集中管理这些会话数据,让各个服务器都可以共享。

 

;