赞
踩
“如何实现Redis延时队列”这个面试题应该也是比较常见的,解答如下:
使用sortedset(有序集合) ,拿时间戳作为 score ,消息内容作为key 调用 zadd 来生产消息,消费者用zrangebyscore 指令获取 N 秒之前的数据轮询进行处理。
Java实现Redis延时队列,首先要了解何为延时队列,即可以将消息存储在队列中,并在指定的延时时间后再将消息出队。这种队列在很多场景下都非常有用,例如消息延时处理,延时确认(订单确认) 等,参考以上解答,思路应该拆分:
首先需要有个延时队列,该队列是通过一定顺序(当前时间戳+延时时间)排序的(即优先取到延时时间已结束的数据),然后消费者端就需要获取到队列中延时时间靠前结束的数据(即当前时间戳+延时时间靠前)。
<dependency>
<groupId>redis.clients</groupId>
<artifactId>jedis</artifactId>
<version>3.7.0</version>
</dependency>
zadd命令用于将一个成员Score值加入到有序集合中。Score值可以是整数或者浮点数。如果有序集合中已经存在相同的成员,那么旧成员将被替代。
语法:ZADD key Score member [Score2 member2 …]
示例:ZADD students 100 alice 或 ZADD students 80 alice 90 bob (添加单个或多个情况)
zrem命令用于从有序集合中移除一个或多个成员。该命令接收两个参数:第一个参数是要操作的有序集合的键,第二个参数是将要移除的成员的值。
语法:ZREM key member [member …]
示例:ZREM students alice
zrangeByScore命令用于获取分数在指定范围内的所有成员。
语法:ZRANGEBYSCORE key min max [WITHSCORES] [LIMIT offset count]
参数说明:
key:Redis中的键。
min:最小分数。
max:最大分数。
WITHSCORES:可选参数,如果设置为true,则返回分数以及成员。
LIMIT:可选参数,用于限制返回的成员数量。
offset:偏移量,从第几个成员开始。
count:限制返回的成员数量。
返回值:
按照分数升序返回成员列表。
如果设置了LIMIT参数,则返回限制数量的成员。
示例:ZADD ZRANGEBYSCORE students 80 90 WITHSCORES
核心部分,消息队列工具类
import redis.clients.jedis.Jedis; import java.util.Set; public class DelayQueueWithRedis { private Jedis jedis; private String queueKey; public DelayQueueWithRedis(Jedis jedis, String queueKey) { this.jedis = jedis; this.queueKey = queueKey; } // 添加消息到延迟队列 public void push(String message, long delaySeconds) { // 计算消息的分数,这里使用消息进入队列的时间加上延迟时间 long score = System.currentTimeMillis() / 1000 + delaySeconds; //向有序集合添加一个成员,并设置其分数 jedis.zadd(queueKey, score, message); } // 获取并消费一条消息 public String pop() { while (true) { long now = System.currentTimeMillis() / 1000; // 只获取分数在0到当前时间的元素 Set<String> messages = jedis.zrangeByScore(queueKey, 0, now, 0, 1); if (messages.isEmpty()) { System.out.println("No messages"); // 没有可消费的消息,休眠一会儿继续尝试 try { Thread.sleep(500); } catch (InterruptedException e) { Thread.currentThread().interrupt(); break; } } else { String message = messages.iterator().next(); // 从有序集合中移除一个成员 jedis.zrem(queueKey, message); return message; } } return null; } }
生产者端测试
import redis.clients.jedis.Jedis; /** * @Author: zhangximing * @Email: 530659058@qq.com * @Date: 2024/2/19 16:53 * @Description: 生产者端测试 */ public class MainP { public static void main(String[] args) { Jedis jedis = new Jedis("localhost",6379); DelayQueueWithRedis delayQueue = new DelayQueueWithRedis(jedis, "delay_queue"); // 添加延时消息 delayQueue.push("message1", 5); delayQueue.push("message2", 10); delayQueue.push("message3", 8); } }
消费者端测试
import redis.clients.jedis.Jedis; /** * @Author: zhangximing * @Email: 530659058@qq.com * @Date: 2024/2/19 16:51 * @Description: 消费者端测试 */ public class MainC { public static void main(String[] args) { Jedis jedis = new Jedis("localhost",6379); DelayQueueWithRedis delayQueue = new DelayQueueWithRedis(jedis, "delay_queue"); // 消费延时消息 while (true) { String message = delayQueue.pop(); if (message != null) { System.out.println("Consumed: " + message); } } } }
测试结果:数据在延时指定时间后才正常打印
Copyright © 2003-2013 www.wpsshop.cn 版权所有,并保留所有权利。