Redis---String
字符串常见命令
符串类型是Redis最基础的数据结构。首先键都是字符串类型。 而且其他几种数据结构都是在字符串类型基础上构建的,所以字符串类型能为其他四种数据结构的学习奠定基础。 字符串类型的值可以是字符串(简单的字符串、复杂的字符串(例如JSON、XML))、数字(整数、浮点数)、甚至是二进制(图片、音频、视频),但是值最大不能超过512MB。
命令 | 作用 | 案例 |
---|---|---|
SET | 将字符串值value关联到key | SET "HELLO" "WORLD" |
APPEND | 将值追加到指定key的值末尾,如果key不存在,则相当于增加操作 | APPEND "HELLO" "!" |
BITCOUNT | 计算给定字符串中,被设置为1 的Bit位的数量 | BITCOUNT "HELLO" |
BITOP | 对一个或多个保存二进制位的字符串key 进行位元操作 | |
DECR | 将 key 中储存的数字值减一。Key不存在,则将值置0,key类型不正确返 回一个错误。(key需要先存在) | DECR A |
DECRBY | 将key所储存的值减去指定数量。 | DECRBY A -1 |
INCR | 将 key 中储存的数字值增一。不能转换为数字则报错 | INCR B |
INCRBY | 将key所储存的值加上指定增量 | INCRBY B 10 |
INCRBYFLOAT | 为key中所储存的值加上指定的浮点数增量(key需要先存在) | INCRBYFLOAT C 0.5 |
GET | 为key中所储存的值加上指定的浮点数增量(key需要先存在) | GET HELLO |
GETBIT | 对key所储存的字符串值,获取指定偏移量上的位 | |
GETRANGE | 返回key中字符串值的子字符串,字符串的截取范围由start和end两个偏移 量决定 | GETRANGE KEY 1 5 |
GETSET | 将给定key的值设为value,并返回key的旧值。非字符串报错。 | GETSET KEY VALUE |
MGET | 返回所有(一个或多个)给定key的值 | MGET KEY1 KEY2 KEY3 |
MSET | 同时设置一个或多个key-value对 | MSET K1 V1 K2 V2 K3 V3 |
MSETNX | 同时设置一个或多个key-value对,若一个key已被占用,则全部的执行取 消。 | MSETNX K1 V1 K2 V2 K3 V3 |
PSETEX | 以毫秒为单位设置key 的有效时间 | PSETEX K4 1000 V4 |
SETEX | 将值value关联到key,并将key的有效时间(秒) | setex k5 1000 v5 |
SETNX | 当key未被使用时,设置为指定值(KEY存在返回0,不存在返回1,可用于分布式锁) | setnx k5 v5 |
STRLEN | 返回key所储存的字符串值的长度 | STRLEN k5 |
SET KEY V XX | 设置value到指定key,key必须存在 | SET KEY V XX |
字符串类型的命令比较多,我们将其分为常用的和不常用的两类。
字符串常见命令
1)设置值
命令:set key value [ex seconds] [px milliseconds] [nx|xx]
set命令的可选参数:
ex seconds:为键设置秒级过期时间。
px milliseconds:为键设置毫秒级过期时间。
nx:键必须不存在,才可以设置成功,用于添加。
xx:与nx相反,键必须存在,才可以设置成功,用于更新。
127.0.0.1:6379> set name kebi OK 127.0.0.1:6379> set name kebi ex 20 xx OK 127.0.0.1:6379> set age 26 nx OK
除了set,Redis还提供了setex和setnx两个命令
语法:
127.0.0.1:6379> set age 26 nx OK 127.0.0.1:6379> setnx age 26 (integer) 0 #设置失败,setnx必须设置不存在的 127.0.0.1:6379> set age 27 OK #可以更新 127.0.0.1:6379> setex name 30 kebi OK #可以更新
setex key seconds value
setnx key values
setex和setnx可以看作set的专业定制版,适合某些场合。
ex是用来设置时间的,所以后面必须接时间。
nx是键必须不存在,所以不能用来更新。
setnx和setex有什么用了?
以setnx命令为例,由于Redis的单线程命令处理机制,如果有多个客户端同时执行setnx key value,
根据setnx的特性只有一个客户能设置成功,setnx可以作为一种分布式锁的一种实现方案。
2)获取值
命令:get key
127.0.0.1:6379> get age "27" 127.0.0.1:6379> get age_kebi (nil) #如果要获取的键不存在,则返回nil(空)。
3)批量设置值
命令:mset key value [key value...]
下面操作通过mset命令一次性设置4个键值对:
127.0.0.1:6379> mset a 1 b 2 c 3 d 4 OK
4)批量获取值
命令:mget [key...]
下面操作批量获取4个键的值:
127.0.0.1:6379> mget a b c d 1) "1" 2) "2" 3) "3" 4) "4" #如果有些键不存在,那么它的值为nil(空),结果是按照传入键的顺序返回:
127.0.0.1:6379> mget a b c f 1) "1" 2) "2" 3) "3" 4) (nil)
批量操作命令可以有效提高开发效率,假如没有mget这样的命令,要执行n次get命令需要按照如下方式来执行
具体耗时:n次get时间 = n次网络时间 + n次命令时间
使用mget命令后,要执行n次get命令操作只需要按照如下方式即可:
具体耗时:n次get时间 = 1次网络时间 + n次命令时间
Redis可以支撑每秒数万的读写操作,但是这指的是Redis服务端的处理能力,
对于客户端来说,一次命令除了命令时间还有网络时间,假设网络时间为1毫秒,命令时间为0.1毫秒,
按照每秒处理一万次命令算,那么执行1000次get命令和一次mget命令的区别在于:
操作 | 时间 |
1000次get | 1000 x 1 + 1000 x 0.1 = 1100毫秒 = 1.1秒 |
1次mget(组装了1000个键值对) | 1 x 1 + 1000 x 0.1 = 101毫秒 = 0.101秒 |
因为Redis的处理能力已经足够高,对于开发人员来说,网络可能成为性能的瓶颈。
学会使用批量操作,有助于提高业务处理效率,但是要注意的是每次批量操作所发送的命令不是无节制的,
如果数量过多可能造成Redis阻塞或者网络阻塞。
计数
命令:incr key
incr命令用于对值做自增操作,返回结果分为三种情况:
(1)值不是整数,返回错误。
(2)值是整数,返回自增后的结果。
(3)键不存在,按照值为0自增,返回结果是1.
#值不是整数 127.0.0.1:6379> set name kebi OK 127.0.0.1:6379> incr name (error) ERR value is not an integer or out of range #报错 #值是整数 127.0.0.1:6379> set age 25 OK 127.0.0.1:6379> incr age (integer) 26 #加1返回结果 127.0.0.1:6379> incr age (integer) 27 #值不存在,返回1 127.0.0.1:6379> exists sex (integer) 0 127.0.0.1:6379> incr sex (integer) 1
命令 decr key
#值为整数
127.0.0.1:6379> incr sex (integer) 7 127.0.0.1:6379> decr sex (integer) 6 127.0.0.1:6379> decr sex (integer) 5 127.0.0.1:6379> exists a (integer) 1 127.0.0.1:6379> decr a (integer) 0 127.0.0.1:6379> decr a #可以返回负数 (integer) -1 127.0.0.1:6379> decr a (integer) -2 127.0.0.1:6379> decr name #不是整数报错 (error) ERR value is not an integer or out of range
命令 incrby key increment 添加指定数字
#值不存在 127.0.0.1:6379> exists g (integer) 0 127.0.0.1:6379> incrby g 3 (integer) 3 127.0.0.1:6379> incrby g 3 (integer) 6 #值不是整数 127.0.0.1:6379> incrby name (error) ERR wrong number of arguments for 'incrby' command #值是整数 127.0.0.1:6379> decr age (integer) 26 127.0.0.1:6379> incrby age 2 (integer) 28 127.0.0.1:6379> incrby age 2 (integer) 30
命令 decrby key decrement 自减指定数字
#值不存在 127.0.0.1:6379> exists h (integer) 0 127.0.0.1:6379> decrby h 2 (integer) -2 #值不是整数 127.0.0.1:6379> decrby name (error) ERR wrong number of arguments for 'decrby' command #值为整数 127.0.0.1:6379> decrby age 3 (integer) 27 127.0.0.1:6379> decrby age 3 (integer) 24
命令 incrbyfloat key increment 添加指定小数
#值不是整数 127.0.0.1:6379> incrbyfloat name 1.5 (error) ERR value is not a valid float #值是整数 127.0.0.1:6379> incrbyfloat age 2.5 "28.5" #值不存在 127.0.0.1:6379> incrbyfloat y 2.5 "2.5"
注意:上面这些不同的类型指的增减数字的类型,例如是每次加1还是1.5
很多存储系统和编程语言内部都使用CAS机制实现技术功能,会有一定的CPU开销,但是在Redis中完全不存在这个问题
因为Redis是单线程架构,任何命令到了Redis服务端都要顺序执行。
字符串不常见命令
1)追加值
命令:append key value
append可以向字符串尾部追加值:
127.0.0.1:6379> get name "kebi" 127.0.0.1:6379> append name goods #追加 (integer) 9 #返回这个值的长度 127.0.0.1:6379> get name "kebigoods"
2)字符串长度
命令:strlen key
127.0.0.1:6379> get name "kebigoods" #一个字母一个字节 127.0.0.1:6379> strlen name (integer) 9
如果字符串是汉字,一个汉字占3个字节:
127.0.0.1:6379> set name:chinese "科比" OK 127.0.0.1:6379> strlen name:chinese (integer) 6
3)设置并返回原值
命令:get key value
getset和set一样能设置值,但不同的是,它会返回原值:
127.0.0.1:6379> getset name kebi-bryant "kebigoods" 127.0.0.1:6379> getset class seven (nil) #原值不存在返回空
4)设置指定位置的字符
命令:setrange key offeset values
127.0.0.1:6379> set drom 301 OK 127.0.0.1:6379> setrange drom 3 B #替换掉指定位置的字符,可以加 (integer) 4 127.0.0.1:6379> get drom "301B" #注意外国人计数从0开始 #替换之后中间有空格,用十六进制补 127.0.0.1:6379> set drom 301 OK 127.0.0.1:6379> setrange drom 4 B (integer) 5 127.0.0.1:6379> get redis (nil) 127.0.0.1:6379> get drom "301\x00B"
5)获取部分字符串
命令:getrange key start end
start和end分别是开始和结束的偏移量,偏移量从0开始计算
127.0.0.1:6379> get name "kebi-bryant" 127.0.0.1:6379> getrange name 0 3 "kebi"
最后提供一张字符串类型命令的时间复杂度,之前讲dbsize命令时提到过,可以作为参考依据。结合自身业务需求和数据大小选择适合的命令。
命令 | 时间复杂度 |
set key value | O(1) |
get key | O(1) |
del key [key ...] | O(k),k是键的个数 |
mset key value [key value ...] | O(k),k是键的个数 |
mget key [key...] | O(k),k是键的个数 |
incr key | O(1) |
decr key | O(1) |
decrby key decrement | O(1) |
incrby key increment | O(1) |
incrbyfloat key increment | O(1) |
append key value | O(1) |
strlen key | O(1) |
setrange key offset value | O(1) |
getrange key start end |
O(n),n是字符串长度,由于获取字符串非常快 所以,字符串不是很长,可以视同为O(1) |