五种数据类型及应用场景
String:分布式锁,mysql批量获取序列号,存储对象
Hash:存储对象(更方便管理,集群中数据过于集中),电商购物车
List:栈,队列,阻塞队列,公众号消息
Set:抽奖,点赞收藏,关注模型
ZSet:微博热点榜单,七日热点榜单
string
字符串常用操作:
- SET key value //存入字符串键值对
- MSET key value [key value …] //批量存储字符串键值对
- SETNX key value //存入一个不存在的字符串键值对
- GET key //获取一个字符串键值
- MGET key [key …] //批量获取字符串键值
- DEL key [key …] //删除一个键
- EXPIRE key seconds //设一个键的过期时间(秒)
原子加减:
- INCR key //将key中储存的数字值加1
- DECR key //将key中储存的数字值减1
- INCRBY key increment //将key所储存的值加上increment
- INCRBY key increment //将key所储存的值减去decrement
string类型应用场景
- 分布式锁:使用SETNX 命令,如果key不存在存入值返回1,如果key存在,不会做任何操作返回0
- 批量获取序列号:MySQL分库分表主键自增可以使用INCRBY orderid 1000,获取1000个序列号,用完再执行获取
- 存储对象:MSET user:01xm XX ,user:01xb 男
hash
hash常用操作
- HSET key field value //存储一个哈希表key的键值
- HSETNX key field value //存储一个不存在的哈希表key的键值
- HMSET kev field value [field value …] //在一个哈希表key中存储多个键值对
- HGET key field //获取哈希表key对应的field键值
- HMGET key field [field …] //批量获取哈希表key中多个fieldt值
- HDEL key field [field …] //删除哈希表key中的field键值
- HLEN key //返回哈希表key中field的数量
- HGETALL key //返回哈希表key中所有的键值
- HINCRBY key field increment //为哈希表key中field健的值加上增量increment
Hash应用场景
作对象缓存:
方便管理,HMSET user 01:name XX 01:sex 男。在Redis集群中,使用hash会使user这个key数据集中在一个槽位上。
电商购物车
1)以用户id为key cart:1001
2)商品id为field 10088
3)商品数量为value 1
购物车操作:
1)添加商品-hset cart:1001 10088 1
2)增加数量→hincrby cart:1001 10088 1
3)商品总数→hlen cart:100
4)删除商品→hdel cart:1001 10088
5)获取购物车所有商品→hgetall cart:1001
List
List常用操作
LPUSH key value [value…]//将一个或多个值value插入到key列表的表头(最左边)
RPUSH key value [value…]//将一个或多个值value插入到key列表的表尾(最右边)
LPOP key//移除并返回key列表的头元素
RPOP key//移除并返回key列表的尾元素
LRANGE key start stop//返回列表key中指定区间内的元素,区间以偏移量start和stop指定
BLPOP key [key …] timeout//从key列表表头弹出一个元素,若列表中没有元素,阻塞等待timeout秒,如果timeout=0,一直阻塞等待
BRPOP key [key …] timeout//从key列表表尾弹出一个元素,若列表中没有元素,阻塞等待timeout秒,如果timeout=0,一直阻塞等待
List应用场景
栈:LPUSH+LPOP
队列:LPUSH+RPOP
阻塞队列:LPUSH+RPOP。执行BRPOP如果队列中有元素,和RPOP效果一样,如果队列中没有元素,会阻塞,等待队列中有值直接取出
微博消息和微信公号消息
诸葛老师关注了MacTalk,备胎说车等大V
1)MacTalk微博,消息ID为10018
LPUSH msg:诸葛老师-ID) 10018
2)备胎说车发微博,消息ID为10086
LPUSH msg:{诸葛老师-ID) 10086 1
3)查看最新微博消息
LRANGE msg:{诸葛老师-ID) 0 5
Set
Set常用操作
SADD key member [member …]//往集合key中存入元素,元素存在则忽略,若key不存在则新建
SREM key member [member …]//从集合key中删除元素
SMEMBERS key//获取集合key中所有元素
SCARD key//获取集合key的元素个数
SISMEMBER key member//判断member元素是否存在于集合key中
SRANDMEMBER key [count]//从集合key中选出count个元素,元素不从key中删除
SPOP key [count]//从集合key中选出count个元素,元素从key中删除
Set运算操作
SINTER key [key …]//交集运算
SINTERSTORE destination key [key…]//将交集结果存入新集合destination中
SUNION key [kev…]/井集运算
SUNIONSTORE destination key [key …]//将并集结果存入新集合destination中
SDIFF key [key …]//差集运算
SDIFFSTORE destination key [key …]//将差集结果存入新集合destination中
Set应用场景
微信抽奖:
1)点击参与抽奖加入集合
SADD key (userID)
2)查看参与抽奖所有用户
SMEMBERS key
3)抽取count名中奖者
SRANDMEMBER key [count] /SPOP key [count]
微信微博点赞,收藏,标签
1)点赞
SADD like:{消息ID} {用户ID}
2)取消点赞
SREM like:{消息ID} {用户ID}
3)检查用户是否点过赞
SISMEMBER like:{消息ID} {用户ID}
4)获取点赞的用户列表
SMEMBERS like:{消息ID}
5)获取点赞用户数
SCARD like:{消息ID}
集合操作实现微博微信关注模型
1)诸葛老师关注的人:
zhugeSet-> {yangguo, sima, luban}
2)杨过老师关注的人:
yangguoset {zhuge, sima, luban.guojia}
3)司马老师关注的人:
simaSet-> {zhuge, yangguo, guojia, luban, xuny}
4)我和杨过老师共同关注:
SINTER zhugeSet yangguoSet–> {sima, luban}
5)我关注的人也关注他(杨过老师):
SISMEMBER simaSet yangquoSISMEMBER lubanSet yangquo
6)我可能认识的人:
SDIFF yangquoSet zhugeSet->{zhuge, guojia}
ZSet
ZSet常用提作
- ZADD key score member [score member]…]//往有序集合key中加入带分值元素
- ZREM key member [member …]//从有序集合key中删除元素
- ZSCORE key member//回有序集合key中元素member的分值
- ZINCRBY key increment member//为有序集合key中元素member的分值加上increment
- ZCARD key//返回有序集合key中元素个数
- ZRANGE key start stop [WITHSCORES]//正序获取有序集合key从start下标到stop下标的元素
- ZREVRANGE key start stop [WITHSCORES]//倒序获取有序集合kev从start下标到stop下标的元素
Zset集合操作
ZUNIONSTORE destkey numkeys key [key …] //并集计算
ZINTERSTORE destkey numkeys key [key …] //交集计算
ZSet应用场景
ZSet集合操作实现排行榜
1)点击新闻
ZINCRBY hotNews:20190819 1守护香港
2)展示当日排行前
ZREVRANGE hotNews:20190819 o 9 WITHSCORES
3)七日搜索榜单计算
ZUNIONSTORE hotNews:20190813-20190819 7hotNews:20190813 hotNews:20190814… hotNews:20190819
4)展示七日排行前十
ZREVRANGE hotNews:20190813-20190819 0 9 WITHSCORES
Redis集群
Redis主节点负责写,从节点可以读。
Redis集群,使用hash槽对Redis分片,从节点只负责备份,不可读。
Redis可以设置key固定在某个槽位上。
Redis避免数据倾斜,Redis集群分槽,热点key存储在一个Redis上,数据倾斜。解决方法:把热点key打散加上随机数,使分配在多台Redis上。
缓存穿透:访问数据库一定不存在的key。
解决方法:缓存空对象。
缓存击穿:一个key失效,同时多个请求访问,缓存不命中,都去查询数据库,对数据库压力非常大。
解决方法:如果缓存未命中,加锁,只有加锁成功的线程去查询数据库,设置缓存。没加锁成功的等待然后重新获取key。
缓存雪崩:大量key同时实效,或者缓存不可用。
解决方法:做Redis集群,在key原有的实效时间上加上一个随机时间,使key过期时间分散。
Redis跳表:在有序链表的基础上,增加多级索引,每隔一个数据提取一个索引,实现快速查找(类似二分查找)。
Redis内存淘汰策略:
全局键三种:报错,随机删除,删除使用最少的
过期键三种:在过期键空间中删除使用最少的/随机删除/有更早过期时间的