Python操作Redis
一、字符串 strings
Python操作Redis的redis模块对字符串(string)的主要操作函数包括:SET、GET、GETSET、SETEX、SETNX、MSET、MSETNX、INCR(INCRBY,DECR,DECRBY在python中庸同一个函数incr实现)、APPEND、SETRANGE、STRLEN。函数说明如下:
- SET: 为指定的键(key)设置值(value), set(self, name, value, **kwargs)。
- GET:获取指定键(key)绑定的值(value),get(self, name)。
- GETSET:为指定的键(key)设置新的值(value),并返回旧的值(old Value),getset(self, name, value)
- SETEX:为指定的键(key)设置过期以秒(second)计的过期时间,setex(self, name, value, time)
- SETNX:键(key)不存在时,为键(key)指定值(value),setnx(self, name, value)
- MSET:一次性设置多个键-值(key-value)对,函数设置的键-值对(即mapping所指内容)数据要以Python字典数据类型传入,mset(self, mapping)
- MSETNX:键-值(key-value)对不存在时,设置键-值(key-value)对,msetnx(self, mapping),mapping值参考6
- INCR:自增函数,默认步长为1,通过对步长(amount)大小以及字符的控制实现了INCRBY(amount>=1)、DECR(amount=-1)、DECRBY(amount<=-1)等函数功能,incr(self, name, amount=1)
- APPEND:为指定的字符串追加值,若不存在则直接创建,append(self, key, value)
- SETRANGE:用 value 参数覆写给定 key 所储存的字符串值,从偏移量 offset 开始,setrange(self, name, offset, value)
- STRLEN:返回字符串的长度,当name不存在时返回0,strlen(self, name)
示例代码如下:
1 pool = redis.ConnectionPool(host='127.0.0.1', port=6379) 2 r = redis.Redis(connection_pool=pool) 3 r.flushall() # 清空Redis 4 5 r.setex('name', value='liaogx', time=2) # 设置新值,过期时间为3s 6 r.mset(k1 = 'v1', k2 = 'v2', k3 = 'v3') # 批量设置新值 7 print(r.mget('k1', 'k2', 'k3', 'k4')) # 批量获取新值 8 print(r.getset('name', 'liaogaoxiang')) # 设置新值并获取原来的值 9 print(r.getrange('name', 0, 1)) # 获取子序列 0 <= x <= 1 10 r.setrange('name', 0, 'LIAO') # 修改字符串内容,从指定字符串索引开始向后替换(新值太长时,则向后添加),返回值的长度 11 i = 0 12 while i < 4: 13 print(r.get('name')) 14 time.sleep(1) 15 i += 1 16 source = 'foo' 17 r.set('n1', source) 18 r.setbit('n1', 7, 1) 19 ''' 20 注:如果在Redis中有一个对应: n1 = "foo", 21 那么字符串foo的二进制表示为:01100110 01101111 01101111 22 所以,如果执行 setbit('n1', 7, 1),则就会将第7位设置为1, 23 那么最终二进制则变成 01100111 01101111 01101111,即:"goo" 24 ''' 25 print(r.get('n1')) 26 print(r.getbit('n1', 7)) # 获取n1对应的值的二进制表示中的某位的值 (0或1) 27 r.set('n2', '李凯') 28 print(r.strlen('n2')) # 返回对应的字节长度(一个汉字3个字节) 29 r.set('num', 1) 30 r.incr('num', amount=10) 31 r.decr('num', amount=1) 32 print(r.get('num')) # 自增num对应的值,当name不存在时,则创建name=amount,否则,则自增。 33 r.append('num', 111) 34 print(r.get('num')) # 在redis num对应的值后面追加内容
输出结果如下:
1 [b'v1', b'v2', b'v3', None] 2 b'liaogx' 3 b'li' 4 b'LIkai' 5 b'LIkai' 6 b'LIlai' 7 b'LIkai' 8 b'goo' 9 1 10 9 11 b'10' 12 b'10111'
二、列表 lists
Python操作Redis主要利用了redis模块来实现,list表操作函数主要模拟了Redis操作命令LPUSH,LRANGE,LINDEX,BLPOP,BRPOP。函数说明如下:
- lpush函数实现了从向指定redis列表头部压入数据功能,
lpush key value
- lrange获取列表指定范围的函数,
lrange key start end
- lindex根据列表下标量获取列表元素值,
lindex key index
- blpop从列表头部取出第一个元素,返回该元素值并从列表删除(l代表left,左边)
- brpop从列表尾部取出第一个元素,返回该元素值并从列表删除(l代表right,右边)
- LPUSHX、RPUSHX以及其他一些Redis列表(List)函数目前版本无法实现,如有可实现方法博客代码事例将会得到更新。
示例代码如下:
1 pool = redis.ConnectionPool(host='127.0.0.1', port=6379) 2 r = redis.Redis(connection_pool=pool) 3 r.flushall() # 清空Redis 4 5 r.lpush('oo', 11) # 保存顺序为: 33,22,11 6 r.lpushx('oo', 00) # 在name对应的list中添加元素,只有name已经存在时,值添加到列表的最左边 7 print(r.llen('oo')) # name对应的list元素的个数 8 9 r.linsert('oo', 'before', 11, 99) # 在11之前插入值99 10 r.lset('oo', 1, 88) # 对name对应的list中的某一个索引位置重新赋值 11 print(r.lrange('oo', 0, -1)) # 在name对应的列表分片获取数据 12 13 r.lrem('oo', 88, num=1) # 在name对应的list中删除指定的值.num=0,删除列表中所有的指定值;num=2,从前到后,删除2个;num=-2,从后向前,删除2个 14 print(r.lrange('oo', 0, -1)) 15 16 print(r.lpop('oo')) # 在name对应的列表的左侧获取第一个元素并在列表中移除,返回值则是第一个元素 17 print(r.lindex('oo', 0)) # 在name对应的列表中根据索引获取列表元素 18 r.lpush('l1', 11) # index为0 19 r.rpush('l1', 22) 20 r.rpush('l1', 33) 21 r.rpush('l1', 44) 22 r.rpush('l1', 55) # index为4 23 r.ltrim('l1', 1, 3) # 在name对应的列表中移除没有在[start-end]索引之间的值 24 print(r.lrange('l1', 0, -1)) 25 26 r.rpoplpush('l1', 'l1') # 从一个列表取出最右边的元素,同时将其添加至另一个列表的最左边;src要取数据的列表的name, dst要添加数据的列表的name 27 print(r.lrange('l1', 0, -1)) 28 29 r.brpoplpush('l1', 'l1', timeout=3) # # timeout,当src对应的列表中没有数据时,阻塞等待其有数据的超时时间(秒),0 表示永远阻塞 30 print(r.lrange('l1', 0, -1)) 31 32 print(r.blpop('l1', 3)) # 从列表头部取出第一个元素,返回该元素值并从列表删除(l代表left,左边) 33 print(r.lrange('l1', 0, -1)) 34 ''' 35 # 由于redis类库中没有提供对列表元素的增量迭代,如果想要循环name对应的列表的所有元素,那么就需要: 36 # 1、获取name对应的所有列表 37 # 2、循环列表 38 # 但是,如果列表非常大,那么就有可能在第一步时就将程序的内容撑爆,所有有必要自定义一个增量迭代的功能: 39 ''' 40 print('自定义增量迭代:') 41 r.flushall() 42 r.lpush('l1', 11) # index为0 43 r.rpush('l1', 22) 44 r.rpush('l1', 33) 45 r.rpush('l1', 44) 46 r.rpush('l1', 55) # index为4 47 48 def list_iter(name): 49 list_count = r.llen(name) 50 for index in range(list_count): 51 yield r.lindex(name, index) 52 53 for item in list_iter('l1'): 54 print(item)
输出结果如下:
1 2 2 [b'0', b'88', b'11'] 3 [b'0', b'11'] 4 b'0' 5 b'11' 6 [b'22', b'33', b'44'] 7 [b'44', b'22', b'33'] 8 [b'33', b'44', b'22'] 9 (b'l1', b'33') 10 [b'44', b'22'] 11 自定义增量迭代: 12 b'11' 13 b'22' 14 b'33' 15 b'44' 16 b'55'
三、集合 sets
Redis 数据库集合对象(set object)是由string类型的无重复元素的无需集合,底层编码可以是intset或者hashtable。intset编码的集合对象用整数集合最为底层实现,所有对象元素保存在整数集合中。Python的redis模块实现了 SADD、SCARD 、SDIFF 、SDIFFSTORE、SINTER 、SINTERSTORE、SISMEMBER 、SMEMBERS 、SMOVE、SPOP、SRANDMEMBER、SREM、SUNION、SUNIONSTORE操作命令的基本用法。函数说明如下:
- SADD:向集合对象添加成员,sadd(self, name, value)
- SCARD:获取集合元素个数,scard(self, name)
- SDIFF:返回给定多个集合对象的差集(set key1-set key2..-set keyn),从左到右计算集合差集,sdiff(self, keys, *args)
- SDIFFSTORE:返回给定给定多个集合对象的差集并存储在目标(dest)集合中,sdiffstore(self, dest, keys, *args)
- SINTER:返回给定所有集合(keys, args)的交集, sinter(self, keys, args)
- SINTERSTORE:返回给定所有集合(keys, args)的交集并存储在 集合(dest) 中, sinterstore(self, dest, keys, args)
- SISMEMBER:判断 value元素是否是集合 name 的成员,返回布尔逻辑值True或者False,sismember(self, name, value)
- SMEMBERS:返回集合中所元素值的Python集合类型数据,smembers(self, name)
- SMOVE:将元素value从集合src移动到 集合 dest。若元素value在集合src中不存在,则集合dest中不会添加元素value, smove(self, src, dest, value)
- SPOP:移除并返回集合中的一个随机元素, spop(self, name)
- SRANDMEMBER:返回集合中一个随机数,但是不做移除操作, srandmember(self, name)。从Redis 2.6 版本开始, 在命令行下Srandmember 命令接受可选返回元素数量的参数 redis>SRANDMEMBER name count
- SREM:移除集合中一个元素,srem(self, name, value),redis模块任然沿用Redis 2.4 版本以前的只接受单个元素的用法。
- SUNION:返回所有给定集合的并集中所有元素,sunion(self, keys, *args)
- SUNIONSTORE:所有给定集合的并集存储在集合dest 中, sunionstore(self, dest, keys, *args)
- SSCAN:迭代集合中的元素,sscan(self, name, cursor=0, match=None, count=None)
示例代码如下:
1 pool = redis.ConnectionPool(host='127.0.0.1', port=6379) 2 r = redis.Redis(connection_pool=pool) 3 r.flushall() # 清空Redis 4 5 r.sadd('s1', 'v1', 'v1', 'v2', 'v3') # name对应的集合中添加元素 6 r.sadd('s2', 'v2', 'v4') # name对应的集合中添加元素 7 print(r.scard('s1')) # 获取name对应的集合中元素个数 8 print(r.sdiff('s1', 's2')) #在第一个name对应的集合中且不在其他name对应的集合的元素集合 9 10 r.sdiffstore('s3', 's1', 's2') # 获取第一个name对应的集合中且不在其他name对应的集合,再将其新加入到dest对应的集合中 11 print(r.smembers('s3')) # 获取s3对应的集合的所有成员 12 13 print(r.sinter('s1', 's2')) # 获取s1, s2对应集合的交集 14 r.sinterstore('s4', 's1', 's2') # 获取s1, s2对应集合的交集,并将其存放到集合是s4中 15 print(r.smembers('s4')) 16 17 print(r.sunion('s1', 's2')) # 获取s1, s2对应集合的并集 18 19 r.sunionstore('s5', 's1', 's2') # 获取s1, s2对应集合的交集,并将其存放到集合是s5中 20 print(r.smembers('s5')) 21 22 print(r.sismember('s4', 'v4')) # 检查value是否是name对应的集合的成员 23 24 r.smove('s2', 's1', 'v4') # 将集合s2中成员v4移至集合s1中 25 print(r.smembers('s1')) 26 r.srem('s1', 'v1') # 在name对应的集合中删除某些值 27 28 print(r.spop('s1')) # 从集合的右侧(尾部)移除一个成员,并将其返回 注意:集合是无序的,故结果随机! 29 30 print(r.srandmember('s1')) # 从name对应的集合中随机获取 numbers 个元素(Redis 2.6+)
输出结果如下:
1 3 2 {b'v3', b'v1'} 3 {b'v3', b'v1'} 4 {b'v2'} 5 {b'v2'} 6 {b'v3', b'v2', b'v4', b'v1'} 7 {b'v3', b'v2', b'v4', b'v1'} 8 False 9 {b'v3', b'v2', b'v4', b'v1'} 10 b'v2' 11 b'v3'
四、有序集合 sorted sets
命令 | 描述 |
---|---|
Redis Zadd 命令 | 向有序集合添加一个或多个成员,或者更新已存在成员的分数 |
Redis Zcard 命令 | 获取有序集合的成员数 |
Redis Zcount 命令 | 计算在有序集合中指定区间分数的成员数 |
Redis Zincrby 命令 | 有序集合中对指定成员的分数加上增量 increment |
Redis Zinterstore 命令 | 计算给定的一个或多个有序集的交集并将结果集存储在新的有序集合 key 中 |
Redis Zlexcount 命令 | 在有序集合中计算指定字典区间内成员数量 |
Redis Zrange 命令 | 通过索引区间返回有序集合成指定区间内的成员 |
Redis Zrangebylex 命令 | 通过字典区间返回有序集合的成员 |
Redis Zrangebyscore 命令 | 通过分数返回有序集合指定区间内的成员 |
Redis Zrank 命令 | 返回有序集合中指定成员的索引 |
Redis Zrem 命令 | 移除有序集合中的一个或多个成员 |
Redis Zremrangebylex 命令 | 移除有序集合中给定的字典区间的所有成员 |
Redis Zremrangebyrank 命令 | 移除有序集合中给定的排名区间的所有成员 |
Redis Zremrangebyscore 命令 | 移除有序集合中给定的分数区间的所有成员 |
Redis Zrevrange 命令 | 返回有序集中指定区间内的成员,通过索引,分数从高到底 |
Redis Zrevrangebyscore 命令 | 返回有序集中指定分数区间内的成员,分数从高到低排序 |
Redis Zrevrank 命令 | 返回有序集合中指定成员的排名,有序集成员按分数值递减(从大到小)排序 |
Redis Zscore 命令 | 返回有序集中,成员的分数值 |
Redis Zunionstore 命令 | 计算给定的一个或多个有序集的并集,并存储在新的 key 中 |
Redis Zscan 命令 | 迭代有序集合中的元素(包括元素成员和元素分值) |
1 pool = redis.ConnectionPool(host='127.0.0.1', port=6379) 2 r = redis.Redis(connection_pool=pool) 3 r.flushall() # 清空Redis 4 5 r.zadd('z1', '11', 1, '22', 2, '33', 3, '44', 4, '55', 5, '66', 6, '66', 7) # 在name对应的有序集合中添加元素 6 print(r.zcard('z1')) # 获取name对应的有序集合元素的数量 7 print(r.zcount('z1', 1, 2)) # 获取name对应的有序集合中分数 在 [min,max] 之间的个数 8 r.zincrby('z1', '11', amount=5) # 自增name对应的有序集合的 name 对应的分数 9 print(r.zrange('z1', 0, -1, desc=False, withscores=True)) # 值11被排序到最后;此处表示按元素的值升序排列 10 11 print(r.zrank('z1', 33)) # 获取某个值在 name对应的有序集合中的排行(从 0 开始) 12 13 r.zrem('z1', '66') # 删除name对应的有序集合中值是values的成员 14 print(r.zrange('z1', 0, -1, desc=False, withscores=True)) 15 16 r.zremrangebyrank('z1', 0, 1) # 根据排行范围删除 17 print(r.zrange('z1', 0, -1, desc=False, withscores=True)) 18 19 r.zremrangebyscore('z1', 4.5, 5.5) # 根据分数范围删除 20 print(r.zrange('z1', 0, -1, desc=False, withscores=True)) 21 22 print(r.zscore('z1', 11)) # 获取name对应有序集合中 value 对应的分数 23 24 r.zadd("zset_name", "a1", 6, "a2", 2, "a3", 5) 25 r.zadd('zset_name1', a1=7, b1=10, b2=5) 26 ''' 27 获取两个有序集合的交集并放入dest集合,如果遇到相同值不同分数,则按照aggregate进行操作 28 aggregate的值为: SUM MIN MAX 29 ''' 30 r.zinterstore('zset_name2', ('zset_name', 'zset_name1'), aggregate='Sum') 31 32 print(r.zrange('zset_name2', 0, -1, desc=False, withscores=True))
输出结果如下:
1 6 2 2 3 [(b'22', 2.0), (b'33', 3.0), (b'44', 4.0), (b'55', 5.0), (b'11', 6.0), (b'66', 6.0)] 4 1 5 [(b'22', 2.0), (b'33', 3.0), (b'44', 4.0), (b'55', 5.0), (b'11', 6.0)] 6 [(b'44', 4.0), (b'55', 5.0), (b'11', 6.0)] 7 [(b'44', 4.0), (b'11', 6.0)] 8 6.0 9 [(b'a1', 13.0)]
五、哈希 hashes
Redis 数据库hash数据类型是一个string类型的key和value的映射表,适用于存储对象。redis 中每个hash可以存储键值对多达40亿。Python的redis模块实现了Redis哈希(Hash)命令行操作的几乎全部命令,包括HDEL、HEXISTS、HGET、HGETALL、HINCRBY、HKEYS、HLEN 、HMGET 、HMSET 、HSET 、HSETNX 、HVALS 、HINCRBYFLOAT等命令。函数说明如下:
- HDEL:删除对应哈希(Hash)表的指定键(key)的字段,hdel(self, name, key)
- HEXISTS:检测哈希(Hash)表对应键(key)字段是否存在,返回布尔逻辑,hexists(self, name, key)
- HGET:获取哈希(Hash)指定键(key)对应的值,hget(self, name, key)
- HGETALL:获取哈希(Hash)表的键-值对(key-value pairs),返回python字典类型数据,hgetall(self, name)
- HINCRBY:为哈希表(Hash)指定键(key)对应的值(key)加上指定的整数数值(int,可为负值)hincrby(self>, name, key, amount=1),Redis 中本操作的值被限制在 64 位(bit)有符号数字。
- HKEYS:返回哈希表(Hash)对应键(key)的数组(Python称之为列表List),hkeys(self, name)
- HLEN: 获取哈希表(Hash)中键-值对(key-value pairs)个数,hlen(self, name)
- HMGET:获取哈希表(Hash)中一个或多个给点字段的值,不存在返回nil(Redis命令行)/None(Python),hmget(self, name, keys),其中keys可以为列表(list)
- HMSET:设置对个键-值对(key-value pairs)到哈希表(Hash)中,python输入值(mapping)为字典(dictionary)类型,hmset(self, name, mapping)
- HSET:为哈希表(Hash)赋值,若键(key)存在值(value)则覆盖,不存在则创建,hset(self, name, key, value)
- HSETNX:为哈希表(Hash)不存值(value)的键(key)赋值,存在操作无效,对应值(value)无变化,hsetnx(self, name, key, value)
- HVALS:返回哈希表(Hash)对应值(value)的列表,hvals(self, name)
- HINCRBYFLOAT:为哈希表 key 中的指定字段的浮点数值加上增量 increment ,hincrbyfloat(self, name, key, amount=1.0)
示例代码如下:
1 pool = redis.ConnectionPool(host='127.0.0.1', port=6379) 2 r = redis.Redis(connection_pool=pool) 3 r.flushall() # 清空Redis 4 5 r.hset('n1', 'k1', 'v1') # hset(name, key, value),name对应的hash中设置一个键值对(不存在,则创建;否则,修改) 6 print(r.hget('n1', 'k1')) 7 r.hmset('n2', {'k1': 'v1', 'k2': 'v2', 'k3': 'v3'}) # hmset(name, mapping),在name对应的hash中批量设置键值对 8 print(r.hmget('n2', 'k2')) 9 print(r.hgetall('n2')) # 获取name对应hash的所有键值 10 print(r.hlen('n2')) # 获取name对应的hash中键值对的个数 11 print(r.hkeys('n2')) # 获取name对应的hash中所有的key的值 12 print(r.hvals('n2')) # 获取name对应的hash中所有的value的值 13 print(r.hexists('n2', 'k4')) # 检查name对应的hash是否存在当前传入的key 14 r.hdel('n2', 'k3') # 将name对应的hash中指定key的键值对删除 15 r.hset('n3', 'k1', 1) 16 r.hincrby('n3', 'k1', amount=1) # hincrby(name, key, amount=1),自增name对应的hash中的指定key的value的值,不存在则创建key=amount 17 print(r.hgetall('n3'))
输出结果如下:
1 b'v1' 2 [b'v2'] 3 {b'k1': b'v1', b'k2': b'v2', b'k3': b'v3'} 4 3 5 [b'k1', b'k2', b'k3'] 6 [b'v1', b'v2', b'v3'] 7 False 8 {b'k1': b'2'}