延时队列如何实现
说明:
rocketmq实现的延时队列只支持特定的延时时间段,1s,5s,10s,…2h,不能支持任意时间段的延时
具体实现:rocketmq发送延时消息时先把消息按照延迟时间段发送到指定的队列中(rocketmq把每种延迟时间段的消息都存放到同一个队列中)然后通过一个定时器进行轮训这些队列,查看消息是否到期,如果到期就把这个消息发送到指定topic的队列中,这样的好处是同一队列中的消息延时时间是一致的,还有一个好处是这个队列中的消息时按照消息到期时间进行递增排序的,说的简单直白就是队列中消息越靠前的到期时间越早
流程图
如果想要深入了解的可以看一下ScheduleMessageService这个类
org.apache.rocketmq.store.schedule.ScheduleMessageService
private final ConcurrentMap<Integer /* level */, Long/* delay timeMillis */> delayLevelTable =
new ConcurrentHashMap<Integer, Long>(32);
private final ConcurrentMap<Integer /* level */, Long/* offset */> offsetTable =
new ConcurrentHashMap<Integer, Long>(32);
日志的声明和一些常量就略过了
这里有两个map
一个是延时等级表 对应延时等级和延时时间
一个是进度表 对应延时等级和进度