Java速记手册

Redis06——Redis到底能用在什么地方(上)

2020-02-29  本文已影响0人  Java面典

之前我们介绍了一些列关于Redis的数据结构、持久化、过期&淘汰策略、集群化等知识点,感兴趣的小伙伴可以在文章的末尾查看往期内容。今天将为大家带来Redis的应用。由于本篇文章较长,所以将拆分为两章来讲解。

除了最基本的KV缓存外,Redis还能用到以下方面。

分布式锁

在分布式应用中,经常会遇到并发问题需要处理,这个时候往往就需要用到分布式锁,而Redis很好的提供了分布式锁的处理。常用的Redis分布式锁有以下两种方式。

setnx

Redis2.8 开始加入了set扩展参数,支持setnx 和 expire可以同时执行,是setnx 和 expire在同一个事务中,避免了expire设置过期时间不成功的问题。

Redlock

除了setnx外,Redis还提供了Redlock提供Redis分布式锁。

要求:Redlock需要多个Redis实例支持,这些Redis实例要求相互独立,没有主从关系。

实现:加锁时,客户端会向超过半数的节点发送setnx指令,只要过半数节点set成功,则加锁成功。释放锁时,需要向所有节点发送del指令。

使用场景:Redlock适用于对高可用要求高的场景,即使一台Redis挂了,也不会影响分布式锁的实现。同时Redlock需要更多的Redis实例,相较于setnx性能有所下降,同时运维代价更高。

消息队列

Redis中实现消息队列常用的list、zset 及Stream几种数据结构可以实现。

list

Redis可以通过list实现异步消息队列。

zset

Redis通过zset可以实现延时队列。

Stream

Redis自5.0起开始支持Stream数据结构,Stream是一个能够支持多播可持久化化的消息队列。

Stream结构.png
  1. xadd: 向Stream中添加消息;
// * 表示服务器自动生成ID,后面分别是 key value
// maxlen 表示设置该stream的最大长度,当消息过长,会丢弃老的消息
127.0.0.1:6379> xadd maxlen 100 javamd * test null 
1582983245000-0         // 返回生成的消息ID,1 表示当前时间生成的第一条消息
  1. xdel: 从Stream中逻辑移除消息(设置标记位,并不影响Stream的真实长度);
127.0.0.1:6379> xdel javamd 1582983245000-1
(integer) 1 
  1. xrange:获取Stream中未标记的消息列表;
127.0.0.1:6379> xrange javamd 1582983245000-0 + // 1582983245000-0为最小消息ID的列表
1) 1) 1582983245000-2
   2) 1) “test”
      2)null
1) 1) 1582983245000-0
   2) 1) “test”
      2)null
127.0.0.1:6379> xrange javamd - 1582983245000-8 // 1582983245000-8为最大消息ID的列表
1) 1) 1582983245000-0
   2) 1) “test”
      2)null
2) 1) 1582983245000-2
   2) 1) “test”
      2)null
  1. xlen: 获取Stream中所有消息的长度;
127.0.0.1:6379> xlen javamd
(integer) 8 // 不包含已标记的消息
  1. del:物理删除整个Steram;
127.0.0.1:6379> del javamd
(integer) 1 // 删除整个Stream
  1. xread:在不定义消费组的情况下进行Stream的独立消费;
127.0.0.1:6379> xread count 2 streams javamd 0-0 // 从Stream头部读取两条消息
1) 1) “javamd”
2) 1) 1) 1582983245000-0
       2) 1) “test”
          2)null
    2) 1) 1582983245000-2
       2) 1) “test”
          2)null

使用xread进行顺序消费时,需要记录当前消费消息的ID。下次调用时,需要将当前消费的消息ID作为参数回传。

xread 可以添加block参数,当参数值大于0时,表示阻塞多少ms,如果相应时间内没有消息到来,返回nil。当参数等于0时,表示永远阻塞,直到新的消息添加到Stream中。

  1. xgroup create: 创建消费组;
127.0.0.1:6379> xgroup create javamd group_javamd1 0-0 // 表示从头部开始消费
OK
127.0.0.1:6379> xgroup create javamd group_javamd2 $ // $ 表示从尾部开始消费,只接受新消息
OK
  1. xreadgroup:通过消费组进行消费。
// > 表示从当前消费组的 last_delivered_id 后面开始读
127.0.0.1:6379> xreadgroup GROUP javamd1 c1 count 1 streams javamd >
1) 1) “javamd”
2) 1) 1) 1582983245000-0
       2) 1) “test”
          2)null

未完待续……

Redis系列推荐

Redis05——Redis Cluster 如何实现分布式集群

Redis04——五分钟明白Redis的哨兵模式

Redis03——Redis是如何删除你的数据的

Redis02——Redis内存数据如何保存到磁盘

Redis01——Redis究竟支持哪些数据结构

上一篇 下一篇

猜你喜欢

热点阅读