将字符串值value关联到key。
如果key已经持有其他值,SET就覆写旧值,无视类型。
# 情况1:对字符串类型的key进行SET
redis> SET apple www.apple.com
OK
redis> GET apple
"www.apple.com"
# 情况2:对非字符串类型的key进行SET
redis> LPUSH greet_list "hello" # 建立一个列表
(integer) 1
redis> TYPE greet_list
list
redis> SET greet_list "yooooooooooooooooo" # 覆盖列表类型
OK
redis> TYPE greet_list
string
将key的值设为value,当且仅当key不存在。
若给定的key已经存在,则SETNX不做任何动作。
SETNX是”SET if Not eXists”(如果不存在,则SET)的简写。
redis> EXISTS job # job不存在
(integer) 0
redis> SETNX job "programmer" # job设置成功
(integer) 1
redis> SETNX job "code-farmer" # job设置失败
(integer) 0
redis> GET job # 没有被覆盖
"programmer"
设计模式(Design pattern): 将SETNX用于加锁(locking)
SETNX可以用作加锁原语(locking primitive)。比如说,要对关键字(key)foo加锁,客户端可以尝试以下方式:
SETNX lock.foo <current Unix time + lock timeout + 1>
如果SETNX返回1,说明客户端已经获得了锁,key设置的unix时间则指定了锁失效的时间。之后客户端可以通过DEL lock.foo来释放锁。
如果SETNX返回0,说明key已经被其他客户端上锁了。如果锁是非阻塞(non blocking lock)的,我们可以选择返回调用,或者进入一个重试循环,直到成功获得锁或重试超时(timeout)。
处理死锁(deadlock)
上面的锁算法有一个问题:如果因为客户端失败、崩溃或其他原因导致没有办法释放锁的话,怎么办?
这种状况可以通过检测发现——因为上锁的key保存的是unix时间戳,假如key值的时间戳小于当前的时间戳,表示锁已经不再有效。
但是,当有多个客户端同时检测一个锁是否过期并尝试释放它的时候,我们不能简单粗暴地删除死锁的key,再用SETNX上锁,因为这时竞争条件(race condition)已经形成了:
幸好,以下算法可以避免以上问题。来看看我们聪明的C4客户端怎么办:
GETSET lock.foo <current Unix timestamp + lock timeout + 1>
将值value关联到key,并将key的生存时间设为seconds(以秒为单位)。
如果key 已经存在,SETEX命令将覆写旧值。
这个命令类似于以下两个命令:
SET key value
EXPIRE key seconds # 设置生存时间
不同之处是,SETEX是一个原子性(atomic)操作,关联值和设置生存时间两个动作会在同一时间内完成,该命令在Redis用作缓存时,非常实用。
# 情况1:key不存在
redis> SETEX cache_user_id 60 10086
OK
redis> GET cache_user_id # 值
"10086"
redis> TTL cache_user_id # 剩余生存时间
(integer) 49
# 情况2:key已经存在,key被覆写
redis> SET cd "timeless"
OK
redis> SETEX cd 3000 "goodbye my love"
OK
redis> GET cd
"goodbye my love"
用value参数覆写(Overwrite)给定key所储存的字符串值,从偏移量offset开始。
不存在的key当作空白字符串处理。
SETRANGE命令会确保字符串足够长以便将value设置在指定的偏移量上,如果给定key原来储存的字符串长度比偏移量小(比如字符串只有5个字符长,但你设置的offset是10),那么原字符和偏移量之间的空白将用零比特(zerobytes,"\x00")来填充。
注意你能使用的最大偏移量是2^29-1(536870911),因为Redis的字符串被限制在512兆(megabytes)内。如果你需要使用比这更大的空间,你得使用多个key。
警告
当生成一个很长的字符串时,Redis需要分配内存空间,该操作有时候可能会造成服务器阻塞(block)。在2010年的Macbook Pro上,设置偏移量为536870911(512MB内存分配),耗费约300毫秒, 设置偏移量为134217728(128MB内存分配),耗费约80毫秒,设置偏移量33554432(32MB内存分配),耗费约30毫秒,设置偏移量为8388608(8MB内存分配),耗费约8毫秒。 注意若首次内存分配成功之后,再对同一个key调用SETRANGE操作,无须再重新内存。
模式
因为有了SETRANGE和GETRANGE命令,你可以将Redis字符串用作具有O(1)随机访问时间的线性数组。这在很多真实用例中都是非常快速且高效的储存方式。
# 情况1:对非空字符串进行SETRANGE
redis> SET greeting "hello world"
OK
redis> SETRANGE greeting 6 "Redis"
(integer) 11
redis> GET greeting
"hello Redis"
# 情况2:对空字符串/不存在的key进行SETRANGE
redis> EXISTS empty_string
(integer) 0
redis> SETRANGE empty_string 5 "Redis!" # 对不存在的key使用SETRANGE
(integer) 11
redis> GET empty_string # 空白处被"\x00"填充
"\x00\x00\x00\x00\x00Redis!"
同时设置一个或多个key-value对。
当发现同名的key存在时,MSET会用新值覆盖旧值,如果你不希望覆盖同名key,请使用MSETNX命令。
MSET是一个原子性(atomic)操作,所有给定key都在同一时间内被设置,某些给定key被更新而另一些给定key没有改变的情况,不可能发生。
redis> MSET date "2011.4.18" time "9.09a.m." weather "sunny"
OK
redis> KEYS * # 确保指定的三个key-value对被插入
1) "time"
2) "weather"
3) "date"
redis> SET google "google.cn" # MSET覆盖旧值的例子
OK
redis> MSET google "google.hk"
OK
redis> GET google
"google.hk"
同时设置一个或多个key-value对,当且仅当key不存在。
即使只有一个key已存在,MSETNX也会拒绝所有传入key的设置操作
MSETNX是原子性的,因此它可以用作设置多个不同key表示不同字段(field)的唯一性逻辑对象(unique logic object),所有字段要么全被设置,要么全不被设置。
# 情况1:对不存在的key进行MSETNX
redis> MSETNX rmdbs "MySQL" nosql "MongoDB" key-value-store "redis"
(integer) 1
# 情况2:对已存在的key进行MSETNX
redis> MSETNX rmdbs "Sqlite" language "python" # rmdbs键已经存在,操作失败
(integer) 0
redis> EXISTS language # 因为操作是原子性的,language没有被设置
(integer) 0
redis> GET rmdbs # rmdbs没有被修改
"MySQL"
redis> MGET rmdbs nosql key-value-store
1) "MySQL"
2) "MongoDB"
3) "redis"
如果key已经存在并且是一个字符串,APPEND命令将value追加到key原来的值之后。
如果key不存在,APPEND就简单地将给定key设为value,就像执行SET key value一样。
# 情况1:对不存在的key执行APPEND
redis> EXISTS myphone # 确保myphone不存在
(integer) 0
redis> APPEND myphone "nokia" # 对不存在的key进行APPEND,等同于SET myphone "nokia"
(integer) 5 # 字符长度
# 情况2:对字符串进行APPEND
redis> APPEND myphone " - 1110"
(integer) 12 # 长度从5个字符增加到12个字符
redis> GET myphone # 查看整个字符串
"nokia - 1110"
返回key所关联的字符串值。
如果key不存在则返回特殊值nil。
假如key储存的值不是字符串类型,返回一个错误,因为GET只能用于处理字符串值。
redis> GET fake_key
(nil)
redis> SET animate "anohana"
OK
redis> GET animate
"anohana"
返回所有(一个或多个)给定key的值。
如果某个指定key不存在,那么返回特殊值nil。因此,该命令永不失败。
redis> MSET name huangz twitter twitter.com/huangz1990 #用MSET一次储存多个值
OK
redis> MGET name twitter
1) "huangz"
2) "twitter.com/huangz1990"
redis> EXISTS fake_key
(integer) 0
redis> MGET name fake_key # 当MGET中有不存在key的情况
1) "huangz"
2) (nil)
返回key中字符串值的子字符串,字符串的截取范围由start和end两个偏移量决定(包括start和end在内)。
负数偏移量表示从字符串最后开始计数,-1表示最后一个字符,-2表示倒数第二个,以此类推。
GETRANGE通过保证子字符串的值域(range)不超过实际字符串的值域来处理超出范围的值域请求。
注解
在<=2.0的版本里,GETRANGE被叫作SUBSTR。
redis> SET greeting "hello, my friend"
OK
redis> GETRANGE greeting 0 4 # 返回索引0-4的字符,包括4。
"hello"
redis> GETRANGE greeting -1 -5 # 不支持回绕操作
""
redis> GETRANGE greeting -3 -1 # 负数索引
"end"
redis> GETRANGE greeting 0 -1 # 从第一个到最后一个
"hello, my friend"
redis> GETRANGE greeting 0 1008611 # 值域范围不超过实际字符串,超过部分自动被符略
"hello, my friend"
将给定key的值设为value,并返回key的旧值。
当key存在但不是字符串类型时,返回一个错误。
redis> EXISTS mail
(integer) 0
redis> GETSET mail xxx@google.com # 因为mail之前不存在,没有旧值,返回nil
(nil)
redis> GETSET mail xxx@yahoo.com # mail被更新,旧值被返回
"xxx@google.com"
设计模式
GETSET可以和INCR组合使用,实现一个有原子性(atomic)复位操作的计数器(counter)。
举例来说,每次当某个事件发生时,进程可能对一个名为mycount的key调用INCR操作,通常我们还要在一个原子时间内同时完成获得计数器的值和将计数器值复位为0两个操作。
可以用命令GETSET mycounter 0来实现这一目标。
redis> INCR mycount
(integer) 11
redis> GETSET mycount 0 # 一个原子内完成GET mycount和SET mycount 0操作
"11"
redis> GET mycount
"0"
返回key所储存的字符串值的长度。
当key储存的不是字符串值时,返回一个错误。
redis> SET mykey "Hello world"
OK
redis> STRLEN mykey
(integer) 11
redis> STRLEN nonexisting # 不存在的key长度视为0
(integer) 0
将key中储存的数字值减一。
如果key不存在,以0为key的初始值,然后执行DECR操作。
如果值包含错误的类型,或字符串类型的值不能表示为数字,那么返回一个错误。
本操作的值限制在64位(bit)有符号数字表示之内。
关于更多递增(increment)/递减(decrement)操作信息,参见INCR命令。
# 情况1:对存在的数字值key进行DECR
redis> SET failure_times 10
OK
redis> DECR failure_times
(integer) 9
# 情况2:对不存在的key值进行DECR
redis> EXISTS count
(integer) 0
redis> DECR count
(integer) -1
# 情况3:对存在但不是数值的key进行DECR
redis> SET company YOUR_CODE_SUCKS.LLC
OK
redis> DECR company
(error) ERR value is not an integer or out of range
将key所储存的值减去减量decrement。
如果key不存在,以0为key的初始值,然后执行DECRBY操作。
如果值包含错误的类型,或字符串类型的值不能表示为数字,那么返回一个错误。
本操作的值限制在64位(bit)有符号数字表示之内。
关于更多递增(increment)/递减(decrement)操作信息,参见INCR命令。
# 情况1:对存在的数值key进行DECRBY
redis> SET count 100
OK
redis> DECRBY count 20
(integer) 80
# 情况2:对不存在的key进行DECRBY
redis> EXISTS pages
(integer) 0
redis> DECRBY pages 10
(integer) -10
将key中储存的数字值增一。
如果key不存在,以0为key的初始值,然后执行INCR操作。
如果值包含错误的类型,或字符串类型的值不能表示为数字,那么返回一个错误。
本操作的值限制在64位(bit)有符号数字表示之内。
注解
这是一个针对字符串的操作,因为Redis没有专用的整数类型,所以key内储存的字符串被解释为十进制64位有符号整数来执行INCR操作。
redis> SET page_view 20
OK
redis> INCR page_view
(integer) 21
redis> GET page_view # 数字值在Redis中以字符串的形式保存
"21"
将key所储存的值加上增量increment。
如果key不存在,以0为key的初始值,然后执行INCRBY命令。
如果值包含错误的类型,或字符串类型的值不能表示为数字,那么返回一个错误。
本操作的值限制在64位(bit)有符号数字表示之内。
关于更多递增(increment)/递减(decrement)操作信息,参见INCR命令。
# 情况1:key存在且是数字值
redis> SET rank 50 # 设置rank为50
OK
redis> INCRBY rank 20 # 给rank加上20
(integer) 70
redis> GET rank
"70"
# 情况2:key不存在
redis> EXISTS counter
(integer) 0
redis> INCRBY counter 30
(integer) 30
redis> GET counter
"30"
# 情况3:key不是数字值
redis> SET book "long long ago..."
OK
redis> INCRBY book 200
(error) ERR value is not an integer or out of range
对key所储存的字符串值,设置或清除指定偏移量上的位(bit)。
位的设置或清除取决于value参数,可以是0也可以是1。
当key不存在时,自动生成一个新的字符串值。
字符串会增长(grown)以确保它可以将value保存在指定的偏移量上。当字符串值增长时,空白位置以0填充。
offset参数必须大于或等于0,小于2^32(bit映射被限制在512MB内)。
redis> SETBIT bit 10086 1
(integer) 0
redis> GETBIT bit 10086
(integer) 1
对key所储存的字符串值,获取指定偏移量上的位(bit)。
当offset比字符串值的长度大,或者key不存在时,返回0。
# 情况1:对不存在的key/不存在的offset进行GETBIT,
# 默认为0
redis> EXISTS bit
(integer) 0
redis> GETBIT bit 10086
(integer) 0
# 情况2:对已存在的offset进行GETBIT
redis> SETBIT bit 10086 1
(integer) 0
redis> GETBIT bit 10086
(integer) 1