Redis学习——数据结构下

Posted jnba

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了Redis学习——数据结构下相关的知识,希望对你有一定的参考价值。

4、集合(集合(set)类型也是用来保存多个的字符串元素,但和列表类型不一样的是,集合中不允许有重复元素,并且集合中的元素是无序的,不能通过索引下标获取元素。)

1、命令

.集合内操作

1、添加元素 sadd key element [element ...]返回结果为添加成功的元素个数。

2、删除元素 srem key element [element ...]返回结果为成功删除元素个数。

3、计算元素个数 scard key

(scard的时间复杂度为O(1),它不会遍历集合所有元素,而是直接用Redis内部的变量)

4、判断元素是否在集合中 sismember key element 如果给定元素element在集合内返回1,反之返回0

5、随机从集合返回指定个数元素 srandmember key [count] [count]是可选参数,如果不写默认为1

6、从集合随机弹出元素 spop key (spop操作可以从集合中随机弹出一个元素,Redis从3.2版本开始,spop也支持[count]参数)

7、获取所有元素 smembers key

(smembers和lrange、hgetall都属于比较重的命令,如果元素过多存在阻塞Redis的可能性,这时候可以使用sscan来完成)

.集合间操作

1、求多个集合的交集 sinter key [key ...]

2、求多个集合的并集 suinon key [key ...]

3、求多个集合的差集 sdiff key [key ...]

4、将交集、并集、差集的结果保存 sinterstore destination key [key ...]

suionstore destination key [key ...]

sdiffstore destination key [key ...]

   (集合间的运算在元素较多的情况下会比较耗时,所以Redis提供了上面三个命令(原命令+store)将集合间交集、并集、差集的结果保存在

destination key中,例如下面操作将user:1:follow和user:2:follow两个集合的交集结果保存在user:1_2:inter中,user:1_2:inter本身也是集合类

sinterstore user:1_2:inter user:1:follow user:2:follow

    )

2、内部编码

·intset(整数集合):当集合中的元素都是整数且元素个数小于set-max-intset-entries配置(默认512个)时,Redis会选用intset来作为集合的内部实现,从而减少内存的使用。

·hashtable(哈希表):当集合类型无法满足intset的条件时,Redis会使用hashtable作为集合的内部实现。

3、使用场景(集合类型比较典型的使用场景是标签(tag)。)

(例如一个用户可能对娱乐、体育比较感兴趣,另一个用户可能对历史、新闻比较感兴趣,这些兴趣

                         点就是标签。有了这些数据就可以得到喜欢同一个标签的人,以及用户的共

                         同喜好的标签,这些数据对于用户体验以及增强用户黏度比较重要。例如一

                         个电子商务的网站会对不同标签的用户做不同类型的推荐,比如对数码产品

                         比较感兴趣的人,在各个页面或者通过邮件的形式给他们推荐最新的数码产

                         品,通常会为网站带来更多的利益)

集合类型的应用场景通常为以下几种

·sadd=Tagging(标签)

·spop/srandmember=Random item(生成随机数,比如抽奖)

·sadd+sinter=Social Graph(社交需求)

5、有序集合

( 有序集合相对于哈希、列表、集合来说会有一点点陌生,但既然叫有序集合,那么它和集合必然有着联系,它保留了集合不能有重复成员的特性,但不同的是,有序集合中的元素可以排序。但是它和列表使用索引下标作为排序依据不同的是,它给每个元素设置一个分数(score)作为排序的依据。有序集合中的元素不能重复,但是score可以重复,就和一个班里的同学学号不能重复,但是考试成绩可以相同。)

1、命令

·集合内

1、添加成员 zadd key score member [score member ...] 返回结果代表成功添加成员的个数。

·Redis3.2为zadd命令添加了nx、xx、ch、incr四个选项:

·nx:member必须不存在,才可以设置成功,用于添加。

·xx:member必须存在,才可以设置成功,用于更新。

·ch:返回此次操作后,有序集合元素和分数发生变化的个数

·incr:对score做增加,相当于后面介绍的zincrby。

·有序集合相比集合提供了排序字段,但是也产生了代价,zadd的时间

复杂度为O(log(n)),sadd的时间复杂度为O(1)。

2、计算成员个数 zcard key

3、计算某个成员的分数 zscore key member

4、计算成员的排名

zrank key member 是从分数从低到高返回排名,zrevrank反之

5、删除成员 zrem key member [member ...] 返回结果为成功删除的个数。

6、增加成员的分数 zincrby key increment member

zincrby user:ranking 9 tom(tom增加了9分)

7、返回指定排名范围的成员

zrange key start end [withscores]

zrevrange key start end [withscores]

(有序集合是按照分值排名的,zrange是从低到高返回,zrevrange反之。)

8、返回指定分数范围的成员

其中zrangebyscore按照分数从低到高返回,zrevrangebyscore反之

9、返回指定分数范围成员个数

zcount key min max

zcount user:ranking 200 221 :返回200到221分的成员的个数

10、删除指定排名内的升序元素 

zremrangebyrank key start end :删除第start到第end名的成员

11、删除指定分数范围的成员  zremrangebyscore key min max

zremrangebyscore user:ranking (250 +inf :将250分以上的成员全部删除

·集合间的操作

1、交集 zinterstore destination numkeys key [key ...] [weights weight [weight ...]][aggregate sum|min|max]

这个命令参数较多,下面分别进行说明:

·destination:交集计算结果保存到这个键。

·numkeys:需要做交集计算键的个数。

·key[key...]:需要做交集计算的键。

·weights weight[weight...]:每个键的权重,在做交集计算时,每个键中

的每个member会将自己分数乘以这个权重,每个键的权重默认是1。

·aggregate sum|min|max:计算成员交集后,分值可以按照sum(和)、min(最小值)、max(最大值)做汇总,默认值是sum。

2、并集 zunionstore destination numkeys key [key ...] [weights weight [weight ...]][aggregate sum|min|max]

(该命令的所有参数和zinterstore是一致的,只不过是做并集计算)

2、内部编码

·ziplist(压缩列表):当有序集合的元素个数小于zset-max-ziplist-entries配置(默认128个),同时每个元素的值都小于zset-max-ziplist-value配

置(默认64字节)时,Redis会用ziplist来作为有序集合的内部实现,ziplist可以有效减少内存的使用。

·skiplist(跳跃表):当ziplist条件不满足时,有序集合会使用skiplist作为内部实现,因为此时ziplist的读写效率会下降。

3、使用场景

有序集合比较典型的使用场景就是排行榜系统。例如视频网站需要对用

户上传的视频做排行榜,榜单的维度可能是多个方面的:按照时间、按照播

放数量、按照获得的赞数。本节使用赞数这个维度,记录每天用户上传视频

的排行榜。

6、键管理

1、单个键管理

针对单个键的命令,前面几节已经介绍过一部分了,例如type、del、object、exists、expire等,下面将介绍剩余的几个重要命令。

1、键重命名 rename key newkey (如果newKey已经存在,值也会被覆盖)

(为了防止被强行rename,Redis提供了renamenx命令,确保只有newKey

不存在时候才被覆盖,例如下面操作renamenx时,newkey=python已经存在,

返回结果是0代表没有完成重命名,所以键java和python的值没变)

  在使用重命名命令时,有两点需要注意:

·由于重命名键期间会执行del命令删除旧的键,如果键对应的值比较大,会存在阻塞Redis的可能性,这点不要忽视。

·如果rename和renamenx中的key和newkey如果是相同的,在Redis3.2和之前版本返回结果略有不同。

2、随机返回一个键 randomkey

3、键过期

除了expire、ttl命令以外,Redis还提供了expireat、pexpire、pexpireat、pttl、persist等一系列命令,下面分别进行说明:

·expire key seconds:键在seconds秒后过期。

·expireat key timestamp:键在秒级时间戳timestamp后过期。

ttl命令和pttl都可以查询键的剩余过期时间,但是pttl精度更高可以达到毫秒级别,有3种返回值:

·大于等于0的整数:键剩余的过期时间(ttl是秒,pttl是毫秒)。

·-1:键没有设置过期时间。

·-2:键不存在。

·pexpire key milliseconds:键在milliseconds毫秒后过期。

·pexpireat key milliseconds-timestamp键在毫秒级时间戳timestamp后过期。

在使用Redis相关过期命令时,需要注意以下几点。

1)如果expire key的键不存在,返回结果为0

2)如果过期时间为负值,键会立即被删除,犹如使用del命令一样

3)persist命令可以将键的过期时间清除

4)对于字符串类型键,执行set命令会去掉过期时间,这个问题很容易在开发中被忽视。

5)Redis不支持二级数据结构(例如哈希、列表)内部元素的过期功能,例如不能对列表类型的一个元素做过期时间设置。

6)setex命令作为set+expire的组合,不但是原子执行,同时减少了一次网络通讯的时间

4、迁移键

迁移键功能非常重要,因为有时候我们只想把部分数据由一个Redis迁移到另一个Redis(例如从生产环境迁移到测试环境),Redis发展历程中提

供了move、dump+restore、migrate三组迁移键的方法,它们的实现方式以及使用的场景不太相同,下面分别介绍

(1)move: move key db

move命令用于在Redis内部进行数据迁移,Redis内部可以有多个数据库,彼此在数据上是相互隔离的,move key db就

是把指定的键从源数据库移动到目标数据库中,多数据库功能不建议在生产环境使用,所以这个命令读者知道即可。

(2)dump+restore

dump key

restore key ttl value

dump+restore可以实现在不同的Redis实例之间进行数据迁移的功能,整个迁移的过程分为两步:

1)在源Redis上,dump命令会将键值序列化,格式采用的是RDB格式。

2)在目标Redis上,restore命令将上面序列化的值进行复原,其中ttl参数代表过期时间,如果ttl=0代表没有过期时间。

有关dump+restore有两点需要注意:

第一,整个迁移过程并非原子性的,而是通过客户端分步完成的。

第二,迁移过程是开启了两个客户端连接,所以dump的结果不是在源Redis和目标Redis之间进行传输

(3)migrate

migrate host port key|"" destination-db timeout [copy] [replace] [keys key [key

migrate命令也是用于在Redis实例间进行数据迁移的,实际上migrate命令就是将dump、restore、del三个命令进行组合,从而简化了操作流程。

migrate命令具有原子性,而且从Redis3.0.6版本以后已经支持迁移多个键的

功能,有效地提高了迁移效率,migrate在10.4节水平扩容中起到重要作用。

2、遍历键(Redis提供了两个命令遍历所有的键,分别是keys和scan)

1.全量遍历键 keys pattern (pattern使用的是glob风格的通配符)

·*代表匹配任意字符。

·代表匹配一个字符。

·[]代表匹配部分字符,例如[1,3]代表匹配1,3,[1-10]代表匹配1到10的任意数字。

·\x用来做转义,例如要匹配星号、问号需要进行转义

   如果Redis包含了大量的键,执行keys命令很可能会造成Redis阻塞,所以一般建议不要在生

   产环境下使用keys命令。但有时候确实有遍历键的需求该怎么办,可以在以下三种情况使用:

·在一个不对外提供服务的Redis从节点上执行,这样不会阻塞到客户端的请求,但是会影响到主从复制。

·如果确认键值总数确实比较少,可以执行该命令。

·使用下面要介绍的scan命令渐进式的遍历所有键,可以有效防止阻塞。

2.渐进式遍历:scan cursor [match pattern] [count number]

Redis从2.8版本后,提供了一个新的命令scan,它能有效的解决keys命令存在的问题。和keys命令执行时会遍历所有键不同,scan采用渐进式遍历

的方式来解决keys命令可能带来的阻塞问题,每次scan命令的时间复杂度是O(1),但是要真正实现keys的功能,需要执行多次scan。

scan cursor [match pattern] [count number]各参数含义:

·cursor是必需参数,实际上cursor是一个游标,第一次遍历从0开始,每次scan遍历完都会返回当前游标的值,直到游标值为0,表示遍历结束。

·match pattern是可选参数,它的作用的是做模式的匹配,这点和keys的模式匹配很像。

·count number是可选参数,它的作用是表明每次要遍历的键个数,默认值是10,此参数可以适当增大。

  除了scan以外,Redis提供了面向哈希类型、集合类型、有序集合的扫

  描遍历命令,解决诸如hgetall、smembers、zrange可能产生的阻塞问题,对

  应的命令分别是hscan、sscan、zscan,它们的用法和scan基本类似。

  )

  **渐进式遍历可以有效的解决keys命令可能产生的阻塞问题,但是scan并

    非完美无瑕,如果在scan的过程中如果有键的变化(增加、删除、修改),

    那么遍历效果可能会碰到如下问题:新增的键可能没有遍历到,遍历出了重

    复的键等情况,也就是说scan并不能保证完整的遍历出来所有的键,这些是我们在开发时需要考虑的。

7、数据库管理(Redis提供了几个面向Redis数据库的操作,它们分别是dbsize、select、flushdb/flushall命令)

1.切换数据库(Redis是用数字作为多个数据库的实现。Redis默认配置中是有16个数据库)

select dbIndex

2.flushdb/flushall

flushdb/flushall命令用于清除数据库,两者的区别的是flushdb只清除当前数据库,flushall会清除所有数据库。

flushdb/flushall命令可以非常方便的清理数据,但是也带来两个问题:

·flushdb/flushall命令会将所有数据清除,一旦误操作后果不堪设想。

·如果当前数据库键值数量比较多,flushdb/flushall存在阻塞Redis的可能性。

 

 

参考资料:《Redis开发与运维》

以上是关于Redis学习——数据结构下的主要内容,如果未能解决你的问题,请参考以下文章

Redis学习——数据结构下

Redis学习

学习Redis系列Redis中字典的实现

Redis 学习

分布式缓存技术redis学习系列——redis高级应用(主从事务与锁持久化)

redis学习——数据持久化