Java基于内存的消息队列实现
先看测试情况
需求背景
需求来源于我写的一个动态任务调度框架:https://gitee.com/hyxl-520/auto-job ,该框架需要实现对运行的任务的日志记录。如何有序、高效的采集获得的日志是一个问题,因此我拟采用基于内存的消息队列。本博客只讨论消息队列实现方式,不讨论动态任务调度框架的使用方式。
实现的功能
- 支持可阻塞的消息生产和消费。
- 支持TTL(即消息可设置过期时长)。
- 支持单播和多播。
- 支持发布监听和过期监听
拟采用的数据结构
根据需求和功能,拟采用Java的LinkedBlockingQueue
,具体原因如下:
-
LinkedBlockingQueue
具有可阻塞、线程安全的优点。 -
LinkedBlockingQueue
具有链式结构,由于要实现TTL,即会经常的删除消息队列中的已过期消息,相比ArrayBlockingQueue
,链式结构删除复杂度更低
完整数据结构如下:
Map<String, MessageQueue<MessageEntry<M>>> messageQueues;
public static class MessageQueue<M> {
private final BlockingQueue<M> messageQueue;
public MessageQueue() {
messageQueue = new LinkedBlockingQueue<>();
}
public MessageQueue(int maxLength) {
if (maxLength <= 0) {
throw new IllegalArgumentException("最大容量应该为非负数");
}
messageQueue = new LinkedBlockingDeque<>(maxLength);
}
public M takeMessageSync() throws InterruptedException {
return messageQueue.take();
}
public M takeMessage() {
return messageQueue.poll();
}
public M readMessage() {
return messageQueue.peek();
}
public M tryReadMessage() {
return messageQueue.element();
}
public boolean removeIf(Predicate<? super M> predicate) {
if (predicate == null) {
return false;
}
return messageQueue.removeIf(predicate);
}
public int length() {
return messageQueue.size();
}
public boolean remove(M message) {
if (message == null) {
return false;
}
return messageQueue.remove(message);
}
public Iterator<M> iterator() {
return messageQueue.iterator();
}
private boolean publishMessageSync(M message) throws InterruptedException {
if (message == null) {
return false;
}
try {
messageQueue.put(message);
return true;
} catch (Exception e) {
e.printStackTrace();
throw e;
}
}
private boolean publishMessage(M message) {
if (message == null) {
return false;
}
try {
return messageQueue.offer(message);
} catch (Exception e) {
e.printStackTrace();
}
return false;
}
}
public static class MessageEntry<M> {
long messageId;
M message;
long expiringTime;
public MessageEntry(long messageId, M message, long expiringTime) {
this.messageId = messageId;
this.message = message;
this.expiringTime = expiringTime;
}
public MessageEntry() {
}
}
Map采用ConcurrentHashMap
,确保线程安全。其中MessageQueue
封装了LinkedBlockingQueue
,作为容器的内部类,其中生产者方法为私有,保证容器可以访问其生产者方法,但是消费者在外部订阅后只能访问消费者的方法。
MessageEntry
是消息条目的封装,除了消息内容,还包含消息ID以及过期时间。
模块设计
IMessageQueueContext
:IMessageQueueContext
是消息容器,所有的消息都将放在这个类里面,其实现了IProducer
和ICounsumer
接口,即提供了生产者和消费者的所有功能。并且通过创建守护线程实现了TTL功能。过期监听策略分为两种(ExpirationListenerPolicy
):
- 单线程遍历所有topic的所有消息
- 每个topic并发遍历
前者适合主题少、总消息数少的情况,后者适合主题较多、消息总数较多的情况。
使用工厂模式,主要配置属性有:
/**
* 默认过期时间:ms,对所有消息设置,-1则表示消息均为永久性消息,除非消费者取出,否则将会一直存在。谨慎使用!
*/
private long defaultExpiringTime = -1;
/**
* 是否允许设置单个消息的过期时间
*/
private boolean isAllowSetEntryExpired = false;
/**
* 允许的最大主题数
*/
private int allowMaxTopicCount = 255;
/**
* 允许每个队列的最大消息数
*/
private int allowMaxMessageCountPerQueue;
/**
* 过期监听器的策略
*/
private ExpirationListenerPolicy listenerPolicy;
通过修改属性值可以构建出单播和多播的生产者消费者。
一个消息容器可供多个生产者和消费者使用。
提供topic过期功能,当消费者订阅后会统计一个订阅数目,当消费者取消订阅时如果当前topic的订阅数和消息堆积数为0,若指定时间内没有生产者生产消息将会移除该主题。
MessageProducer
:MessageProducer
为生产者,实现了IProducer
接口(其实最终实现来自于IMessageQueueContext
类),其提供如图上的方法。构建时需要指定所使用的消息容器。
MessageConsumer
:MessageConsumer
为消费者,实现了IConsumer
接口(其实最终实现来自于IMessageQueueContext
类)。对于消息消费强烈建议不要使用如下参数列表String topic
的方法,这个方法将不能使用消息容器提供的topic过期功能,建议先订阅指定队列,再从该队列获取消息,生产者对于队列订阅也提供了阻塞功能。
拓展
消息容器的队列可采用多种实现,如采用DelayQueue
可以实现死信队列。
完整源代码
IMessageQueueContext
import java.util.List;
/**
* 消息上下文抽象接口
*
* @Auther Huang Yongxiang
* @Date 2022/03/20 9:57
*/
public interface IMessageQueueContext<M> {
int length(String topic);
/**
* 使得消息立即过期
*
* @param topic 主题
* @return void
* @throws ErrorExpiredException 过期时发生异常抛出
* @author Huang Yongxiang
* @date 2022/3/20 11:31
*/
void expire(String topic, MessageEntry<M> messageEntry) throws ErrorExpiredException;
/**
* 摧毁消息容器并启动垃圾清理
*
* @return void
* @author Huang Yongxiang
* @date 2022/3/22 14:33
*/
void destroy();
/**
* 添加一个消息发布监听器,监听器的操作不会影响消息的正常发布
*
* @param topic 要添加监听器的主题
* @param listener 监听器
* @return void
* @author Huang Yongxiang
* @date 2022/11/7 9:43
*/
void addMessagePublishedListener(String topic, MessagePublishedListener<M> listener);
/**
* 移除主题下的所有消息发布监听器
*
* @param topic 要移除的主题
* @return java.util.List<com.example.autojob.skeleton.framework.mq.MessagePublishedListener < M>>
* 移除的监听器列表,如果移除失败或不存在,返回空列表
* @author Huang Yongxiang
* @date 2022/11/7 9:53
*/
List<MessagePublishedListener<M>> removeAllMessagePublishedListener(String topic);
boolean removeMessagePublishedListener(String topic, String listenerName);
/**
* 添加一个消息过期监听器
*
* @param topic 主题
* @param listener 监听器
* @return void
* @author Huang Yongxiang
* @date 2022/11/7 9:54
*/
void addMessageExpiredListener(String topic, MessageExpiredListener<M> listener);
/**
* 移除所有的消息过期监听器
*
* @param topic 主题
* @return java.util.List<com.example.autojob.skeleton.framework.mq.MessageExpiredListener < M>>
* @author Huang Yongxiang
* @date 2022/11/7 9:55
*/
List<MessageExpiredListener<M>> removeAllMessageExpiredListener(String topic);
boolean removeMessageExpiredListener(String topic, String listenerName);
}
IConsumer
import java.util.List;
import java.util.concurrent.TimeUnit;
/**
* 消费者抽象接口
*
* @Auther Huang Yongxiang
* @Date 2022/03/21 14:20
*/
public interface IConsumer<M> {
/**
* 阻塞的获取一条消息,可以决定是否将该消息取出,即移出队列
*
* @param topic 主题
* @param isTakeout 是否移出队列,当为false时该方法将退化成非阻塞的
* @return M
* @author Huang Yongxiang
* @date 2022/3/20 10:55
*/
M takeMessageBlock(final String topic, final boolean isTakeout);
M takeMessageNoBlock(final String topic, final boolean isTakeout);
List<M> takeAllMessageNoBlock(final String topic, final boolean isTakeout);
/**
* 取出符合正则表达式的所有主题的第一条消息,该方法为阻塞方法,某个主题如果不存在消息将会阻塞等待
*
* @param regexTopic 正则表达式
* @param isTakeout 是否取出消息
* @return java.util.List<M>
* @author Huang Yongxiang
* @date 2022/11/7 9:58
*/
List<M> takeMessageByRegexTopicBlock(final String regexTopic, final boolean isTakeout);
List<M> takeMessageByRegexTopicNoBlock(final String regexTopic, final boolean isTakeout);
MessageQueue<MessageEntry<M>> subscriptionMessage(String topic);
/**
* 阻塞的尝试订阅指定消息队列,如果存在则立即返回,否则将会等待指定时长,若期间创建则会立即返回,否则等
* 到结束返回null
*
* @param topic 主题
* @param wait 要阻塞获取的时长
* @param unit wait的时间单位
* @return com.example.autojob.skeleton.framework.mq.MessageQueue<com.example.autojob.skeleton.framework.mq.MessageEntry < M>>
* @author Huang Yongxiang
* @date 2022/3/22 14:32
*/
MessageQueue<MessageEntry<M>> subscriptionMessage(String topic, long wait, TimeUnit unit);
void unsubscribe(String topic);
void unsubscribe(String topic, long wait, TimeUnit unit);
}
IProducer
import java.util.concurrent.BlockingDeque;
import java.util.concurrent.BlockingQueue;
import java.util.concurrent.TimeUnit;
/**
* 消息消费者抽象接口
*
* @Auther Huang Yongxiang
* @Date 2022/03/21 14:19
*/
public interface IProducer<M> {
/**
* 注册一个消息队列
*
* @param topic 主题名
* @return boolean
* @author Huang Yongxiang
* @date 2022/3/20 9:59
*/
boolean registerMessageQueue(String topic);
/**
* 注册一个消息队列,队列的数据结构由参数指定
*
* @param topic 主题
* @param queue 队列数据结构类型
* @return boolean
* @author JingGe(* ^ ▽ ^ *)
* @date 2023/5/26 10:05
*/
boolean registerMessageQueue(String topic, BlockingQueue<MessageEntry<M>> queue);
boolean hasTopic(String topic);
boolean hasRegexTopic(String regexTopic);
boolean removeMessageQueue(String topic);
/**
* 非阻塞的发布一条消息,当容量已满时立即返回
*
* @param message 消息内容
* @param topic 队列主题
* @return boolean
* @author Huang Yongxiang
* @date 2022/3/20 9:59
*/
boolean publishMessageNoBlock(final M message, final String topic);
/**
* 非阻塞的发布一条消息,同时指定其过期时间,当容量已满时立即返回
*
* @param message 消息内容
* @param topic 主题
* @param expiringTime 过期时长
* @param unit 时间单位
* @return boolean
* @author Huang Yongxiang
* @date 2022/3/20 10:09
*/
boolean publishMessageNoBlock(final M message, final String topic, final long expiringTime, final TimeUnit unit);
/**
* 阻塞的发布一条消息,当容量已满时等待空出
*
* @param message 消息内容
* @param topic 队列主题
* @return boolean
* @author Huang Yongxiang
* @date 2022/3/20 10:03
*/
boolean publishMessageBlock(final M message, final String topic);
boolean publishMessageBlock(final M message, final String topic, final long expiringTime, final TimeUnit unit);
}
ErrorExpiredException
/**
* 消息过期异常类
*
* @Auther Huang Yongxiang
* @Date 2022/03/20 11:32
*/
public class ErrorExpiredException extends RuntimeException{
public ErrorExpiredException() {
super();
}
public ErrorExpiredException(String message) {
super(message);
}
public ErrorExpiredException(String message, Throwable cause) {
super(message, cause);
}
public ErrorExpiredException(Throwable cause) {
super(cause);
}
protected ErrorExpiredException(String message, Throwable cause, boolean enableSuppression, boolean writableStackTrace) {
super(message, cause, enableSuppression, writableStackTrace);
}
}
ExpirationListenerPolicy
/**
* 过期策略
*
* @Auther Huang Yongxiang
* @Date 2022/03/21 9:30
*/
public enum ExpirationListenerPolicy {
/**
* 单线程监听过期
*/
SINGLE_THREAD,
/**
* 按照主题并发监听过期,总消息数目过多时采取该方式可以使得效率更高,注意topic过多会占用大量线程资源
*/
TOPIC_CONCURRENCY
}
MessageConsumer
import java.util.List;
import java.util.concurrent.TimeUnit;
/**
* 消费者
*
* @Auther Huang Yongxiang
* @Date 2022/03/21 12:53
*/
public class MessageConsumer<M> implements IConsumer<M> {
private MessageQueueContext<M> messageQueueContext;
public MessageConsumer(MessageQueueContext<M> messageQueueContext) {
this.messageQueueContext = messageQueueContext;
}
@Override
public M takeMessageBlock(String topic, boolean isTakeout) {
if (messageQueueContext == null) {
throw new NullPointerException("消息容器为空");
}
return messageQueueContext.takeMessageBlock(topic, isTakeout);
}
@Override
public M takeMessageNoBlock(String topic, boolean isTakeout) {
if (messageQueueContext == null) {
throw new NullPointerException("消息容器为空");
}
return messageQueueContext.takeMessageNoBlock(topic, isTakeout);
}
@Override
public List<M> takeAllMessageNoBlock(String topic, boolean isTakeout) {
return messageQueueContext.takeAllMessageNoBlock(topic, isTakeout);
}
@Override
public List<M> takeMessageByRegexTopicBlock(String regexTopic, boolean isTakeout) {
return messageQueueContext.takeMessageByRegexTopicBlock(regexTopic, isTakeout);
}
@Override
public List<M> takeMessageByRegexTopicNoBlock(String regexTopic, boolean isTakeout) {
return messageQueueContext.takeMessageByRegexTopicNoBlock(regexTopic, isTakeout);
}
@Override
public MessageQueue<MessageEntry<M>> subscriptionMessage(String topic) {
return messageQueueContext.subscriptionMessage(topic);
}
@Override
public MessageQueue<MessageEntry<M>> subscriptionMessage(String topic, long wait, TimeUnit unit) {
return messageQueueContext.subscriptionMessage(topic, wait, unit);
}
@Override
public void unsubscribe(String topic) {
messageQueueContext.unsubscribe(topic);
}
@Override
public void unsubscribe(String topic, long wait, TimeUnit unit) {
messageQueueContext.unsubscribe(topic, wait, unit);
}
public MessageQueueContext<M> getMessageQueueContext() {
return messageQueueContext;
}
public MessageConsumer<M> setMessageQueueContext(MessageQueueContext<M> messageQueueContext) {
this.messageQueueContext = messageQueueContext;
return this;
}
}
MessageProducer
import lombok.Data;
import java.util.concurrent.BlockingQueue;
import java.util.concurrent.TimeUnit;
/**
* 生产者
*
* @Auther Huang Yongxiang
* @Date 2022/03/21 12:54
*/
@Data
public class MessageProducer<M> implements IProducer<M> {
private MessageQueueContext<M> queueContext;
public MessageProducer(final MessageQueueContext<M> messageQueueContext) {
this.queueContext = messageQueueContext;
}
public MessageQueueContext<M> getMessageQueueContext() {
return queueContext;
}
@Override
public boolean registerMessageQueue(String topic) {
return queueContext.registerMessageQueue(topic);
}
@Override
public boolean registerMessageQueue(String topic, BlockingQueue<MessageEntry<M>> queue) {
return queueContext.registerMessageQueue(topic, queue);
}
@Override
public boolean hasTopic(String topic) {
return queueContext.hasTopic(topic);
}
@Override
public boolean hasRegexTopic(String regexTopic) {
return queueContext.hasRegexTopic(regexTopic);
}
@Override
public boolean removeMessageQueue(String topic) {
return queueContext.removeMessageQueue(topic);
}
@Override
public boolean publishMessageNoBlock(M message, String topic) {
return queueContext.publishMessageNoBlock(message, topic);
}
@Override
public boolean publishMessageNoBlock(M message, String topic, long expiringTime, TimeUnit unit) {
return queueContext.publishMessageNoBlock(message, topic, expiringTime, unit);
}
@Override
public boolean publishMessageBlock(M message, String topic) {
return queueContext.publishMessageBlock(message, topic);
}
@Override
public boolean publishMessageBlock(M message, String topic, long expiringTime, TimeUnit unit) {
return queueContext.publishMessageBlock(message, topic, expiringTime, unit);
}
public MessageProducer<M> setMessageQueueContext(MessageQueueContext<M> queueContext) {
this.queueContext = queueContext;
return this;
}
}
MessageQueueContext
import util.id.IdGenerator;
import util.RegexUtil;
import util.StringUtils;
import util.thread.NamedThreadFactory;
import util.thread.ScheduleTaskUtil;
import util.thread.SyncHelper;
import lombok.Setter;
import lombok.experimental.Accessors;
import lombok.extern.slf4j.Slf4j;
import javax.annotation.PostConstruct;
import java.util.*;
import java.util.concurrent.*;
import java.util.concurrent.atomic.AtomicLong;
import java.util.stream.Collectors;
/**
* 内存消息队列的context
*
* @Auther Huang Yongxiang
* @Date 2022/03/18 17:14
*/
@Slf4j
public class MessageQueueContext<M> implements IMessageQueueContext<M>, IProducer<M>, IConsumer<M> {
/**
* 默认过期时间:ms,对所有消息设置,-1则表示消息均为永久性消息,除非消费者取出,否则将会一直存在。谨慎使用!
*/
private long defaultExpiringTime = -1;
/**
* 是否允许设置单个消息的过期时间
*/
private boolean isAllowSetEntryExpired = false;
/**
* 允许的最大主题数
*/
private int allowMaxTopicCount = 255;
/**
* 允许每个队列的最大消息数
*/
private int allowMaxMessageCountPerQueue;
/**
* 过期监听器的策略
*/
private ExpirationListenerPolicy listenerPolicy;
//存储消息的数据结构
private Map<String, MessageQueue<MessageEntry<M>>> messageQueues;
private final Map<String, List<MessagePublishedListener<M>>> publishedListenersMap = new ConcurrentHashMap<>();
private final Map<String, List<MessageExpiredListener<M>>> expiredListenersMap = new ConcurrentHashMap<>();
private boolean isOpenListener = false;
/**
* 守护线程
*/
private ScheduledExecutorService executorService;
/**
* 各个主题的订阅数
*/
private Map<String, AtomicLong> subscriptionCount;
private ThreadPoolExecutor concurrencyThreads;
public static Builder<Object> builder() {
return new Builder<>();
}
private MessageQueueContext() {
}
@PostConstruct
public void init() {
}
@Override
public boolean registerMessageQueue(String topic) {
return registerMessageQueue(topic, new LinkedBlockingQueue<>(allowMaxMessageCountPerQueue));
}
@Override
public boolean registerMessageQueue(String topic, BlockingQueue<MessageEntry<M>> queue) {
if (StringUtils.isEmpty(topic)) {
log.error("创建队列失败,主题为空");
return false;
}
if (messageQueues.containsKey(topic)) {
return false;
}
if (messageQueues.size() >= allowMaxTopicCount) {
log.error("当前消息容器最大支持{}个主题", allowMaxTopicCount);
return false;
}
try {
MessageQueue<MessageEntry<M>> messageQueue = new MessageQueue<>(queue);
messageQueues.put(topic, messageQueue);
if (!isOpenListener) {
synchronized (MessageQueueContext.class) {
if (!isOpenListener) {
if (listenerPolicy == ExpirationListenerPolicy.SINGLE_THREAD) {
scheduleExpiringCheckSingleThread();
} else {
scheduleExpiringCheckTopicConcurrency();
}
isOpenListener = true;
}
}
}
if (subscriptionCount == null) {
subscriptionCount = new ConcurrentHashMap<>();
}
subscriptionCount.put(topic, new AtomicLong(0));
return true;
} catch (Exception e) {
log.error("创建队列发生异常:{}", e.getMessage());
}
return false;
}
@Override
public boolean hasTopic(String topic) {
return messageQueues.containsKey(topic);
}
@Override
public boolean hasRegexTopic(String regexTopic) {
return messageQueues
.keySet()
.stream()
.anyMatch(topic -> RegexUtil.isMatch(topic, regexTopic));
}
@Override
public boolean removeMessageQueue(String topic) {
try {
messageQueues.remove(topic);
return true;
} catch (Exception e) {
log.error("移除消息队列时发生异常:{}", e.getMessage());
}
return false;
}
@Override
public boolean publishMessageNoBlock(M message, String topic, long expiringTime, TimeUnit unit) {
if (!isAllowSetEntryExpired) {
log.error("不允许设置单个消息的过期时间");
return false;
}
if (!messageQueues.containsKey(topic)) {
log.error("发布非阻塞消息失败,所要发布到的队列:{}不存在", topic);
return false;
}
if (expiringTime <= 0 || unit == null) {
log.error("非法过期时间");
return false;
}
if (message == null) {
log.error("禁止发布空消息");
return false;
}
MessageEntry<M> messageEntry = new MessageEntry<>();
messageEntry.setMessageId(IdGenerator.getNextIdAsLong());
messageEntry.setMessage(message);
messageEntry.setExpiringTime(unit.toMillis(expiringTime) + System.currentTimeMillis());
if (messageQueues
.get(topic)
.publishMessage(messageEntry)) {
executeMessagePublishedListeners(topic, message);
return true;
}
return false;
}
@Override
public boolean publishMessageBlock(M message, String topic, long expiringTime, TimeUnit unit) {
if (!isAllowSetEntryExpired) {
log.error("不允许设置单个消息的过期时间");
return false;
}
if (!messageQueues.containsKey(topic)) {
log.error("发布非阻塞消息失败,所要发布到的队列:{}不存在", topic);
return false;
}
if (expiringTime <= 0 || unit == null) {
log.error("非法过期时间");
return false;
}
if (message == null) {
log.error("禁止发布空消息");
return false;
}
MessageEntry<M> messageEntry = new MessageEntry<>();
messageEntry.setMessageId(IdGenerator.getNextIdAsLong());
messageEntry.setMessage(message);
messageEntry.setExpiringTime(unit.toMillis(expiringTime) + System.currentTimeMillis());
try {
if (messageQueues
.get(topic)
.publishMessageSync(messageEntry)) {
executeMessagePublishedListeners(topic, message);
return true;
}
} catch (InterruptedException e) {
log.warn("发布可阻塞消息发生异常,等待时被异常占用:{}", e.getMessage());
}
return false;
}
/**
* 阻塞的获取一条消息,可以决定是否将该消息取出,即移出队列,当多播时最好不要移出队列
*
* @param topic 主题
* @param isTakeout 是否移出队列,当为false时该方法将退化成非阻塞的
* @return M
* @author Huang Yongxiang
* @date 2022/3/20 10:55
*/
@Override
public M takeMessageBlock(String topic, boolean isTakeout) {
if (!messageQueues.containsKey(topic)) {
return null;
}
if (isTakeout) {
try {
return messageQueues
.get(topic)
.takeMessageSync().message;
} catch (InterruptedException e) {
log.warn("可阻塞获取消息发生异常,等待时被异常占用:{}", e.getMessage());
}
return null;
}
return messageQueues
.get(topic)
.readMessage().message;
}
@Override
public M takeMessageNoBlock(String topic, boolean isTakeout) {
if (!hasTopic(topic)) {
return null;
}
if (messageQueues
.get(topic)
.length() == 0) {
return null;
}
if (isTakeout) {
return messageQueues
.get(topic)
.takeMessage().message;
}
return messageQueues
.get(topic)
.readMessage().message;
}
@Override
public List<M> takeAllMessageNoBlock(String topic, boolean isTakeout) {
if (!hasTopic(topic)) {
return Collections.emptyList();
}
if (messageQueues
.get(topic)
.length() == 0) {
return Collections.emptyList();
}
List<M> messages = new LinkedList<>();
try {
if (!isTakeout) {
return messageQueues.get(topic).messageQueue
.stream()
.map(item -> item.message)
.collect(Collectors.toList());
} else {
while (true) {
M message = takeMessageNoBlock(topic, true);
if (message != null) {
messages.add(message);
} else {
break;
}
}
}
} catch (Exception e) {
log.error("获取全部消息时发生异常:{}", e.getMessage());
return Collections.emptyList();
}
return messages;
}
@Override
public List<M> takeMessageByRegexTopicBlock(String regexTopic, boolean isTakeout) {
if (!hasRegexTopic(regexTopic)) {
return null;
}
List<String> topics = messageQueues
.keySet()
.stream()
.filter(topic -> RegexUtil.isMatch(topic, regexTopic))
.collect(Collectors.toList());
log.debug("获取到正则主题:{}的匹配主题{}个", regexTopic, topics.size());
if (topics.size() == 0) {
return Collections.emptyList();
}
List<M> messages = new LinkedList<>();
for (String topic : topics) {
messages.add(takeMessageBlock(topic, isTakeout));
}
return messages;
}
@Override
public List<M> takeMessageByRegexTopicNoBlock(String regexTopic, boolean isTakeout) {
if (!hasRegexTopic(regexTopic)) {
return null;
}
List<String> topics = messageQueues
.keySet()
.stream()
.filter(topic -> RegexUtil.isMatch(topic, regexTopic))
.collect(Collectors.toList());
log.debug("获取到正则主题:{}的匹配主题{}个", regexTopic, topics.size());
if (topics.size() == 0) {
return null;
}
List<M> messages = new LinkedList<>();
for (String topic : topics) {
M m = takeMessageNoBlock(topic, isTakeout);
if (m == null) {
continue;
}
messages.add(m);
}
return messages;
}
@Override
public MessageQueue<MessageEntry<M>> subscriptionMessage(String topic) {
MessageQueue<MessageEntry<M>> messageQueue = messageQueues.get(topic);
if (messageQueue != null && subscriptionCount != null) {
AtomicLong atomicLong = subscriptionCount.get(topic);
atomicLong.incrementAndGet();
}
return messageQueue;
}
@Override
@SuppressWarnings("unchecked")
public MessageQueue<MessageEntry<M>> subscriptionMessage(String topic, long wait, TimeUnit unit) {
MessageQueue<MessageEntry<M>> messageQueue = subscriptionMessage(topic);
if (messageQueue != null) {
return messageQueue;
}
//进行阻塞获取
ScheduledFuture<Object> future = ScheduleTaskUtil
.build(false, "subscriptionBlock")
.EOneTimeTask(() -> {
long blockTime = unit.toMillis(wait);
int i = 0;
try {
do {
if (hasTopic(topic)) {
return messageQueues.get(topic);
}
Thread.sleep(1);
} while (i++ <= blockTime);
return null;
} catch (Exception e) {
log.error("阻塞订阅时发生异常:{}", e.getMessage());
}
return null;
}, 1, TimeUnit.MILLISECONDS);
try {
return (MessageQueue<MessageEntry<M>>) future.get();
} catch (InterruptedException | ExecutionException e) {
log.error("阻塞订阅时发生异常:{}", e.getMessage());
}
return null;
}
@Override
public void unsubscribe(String topic) {
unsubscribe(topic, 5, TimeUnit.SECONDS);
}
@Override
public void unsubscribe(String topic, long wait, TimeUnit unit) {
if (subscriptionCount != null) {
AtomicLong atomicLong = subscriptionCount.get(topic);
atomicLong.decrementAndGet();
if (atomicLong.get() < 0) {
atomicLong.set(0);
}
//当有队列取消订阅,且目前消息数为0,则对指定队列监视5秒,5秒后依然没有生产者发布消息则直接移除主题
if (atomicLong.get() == 0 && length(topic) == 0) {
long w = unit.toMillis(wait);
log.debug("主题为{}的消息队列目前订阅数为0且积压消息为0,当{}ms后若无生产者发布消息将自动删除该主题队列", topic, w);
Thread thread = new Thread(() -> {
try {
int i = 0;
boolean flag = true;
do {
if (length(topic) > 0) {
flag = false;
break;
}
SyncHelper.sleepQuietly(1, TimeUnit.MILLISECONDS);
} while (i++ <= w);
if (flag) {
log.debug("主题:{}自动删除完成", topic);
removeMessageQueue(topic);
}
} catch (Exception e) {
e.printStackTrace();
}
});
thread.setDaemon(true);
thread.setName("unsubListener");
thread.start();
}
}
}
@Override
public int length(String topic) {
if (messageQueues.containsKey(topic)) {
return messageQueues
.get(topic)
.length();
}
return 0;
}
@Override
public void expire(String topic, MessageEntry<M> messageEntry) throws ErrorExpiredException {
if (messageEntry == null || !messageQueues.containsKey(topic)) {
throw new IllegalArgumentException("参数有误,ID非法或主题不存在");
}
try {
boolean flag = messageQueues
.get(topic)
.remove(messageEntry);
if (!flag) {
throw new ErrorExpiredException("移出失败");
}
} catch (Exception e) {
log.error("过期时发生异常");
throw new ErrorExpiredException(e.getMessage());
}
}
@Override
public void destroy() {
messageQueues = null;
if (isOpenListener) {
try {
executorService.shutdown();
isOpenListener = false;
} catch (Exception e) {
log.error("关闭守护线程发生异常:{}", e.getMessage());
}
}
System.gc();
}
@Override
public void addMessagePublishedListener(String topic, MessagePublishedListener<M> listener) {
if (listener == null) {
throw new NullPointerException();
}
if (hasTopic(topic)) {
if (!this.publishedListenersMap.containsKey(topic)) {
synchronized (publishedListenersMap) {
if (!this.publishedListenersMap.containsKey(topic)) {
this.publishedListenersMap.put(topic, new ArrayList<>());
}
}
}
publishedListenersMap
.get(topic)
.add(listener);
} else {
throw new IllegalArgumentException("不存在相关主题:" + topic);
}
}
@Override
public List<MessagePublishedListener<M>> removeAllMessagePublishedListener(String topic) {
if (publishedListenersMap.containsKey(topic)) {
return publishedListenersMap.remove(topic);
}
return null;
}
@Override
public boolean removeMessagePublishedListener(String topic, String listenerName) {
if (publishedListenersMap.containsKey(topic) && !StringUtils.isEmpty(listenerName)) {
return publishedListenersMap
.get(topic)
.removeIf(listener -> listener
.listenerName()
.equals(listenerName));
}
return false;
}
@Override
public void addMessageExpiredListener(String topic, MessageExpiredListener<M> listener) {
if (listener == null) {
throw new NullPointerException();
}
if (hasTopic(topic)) {
if (!this.expiredListenersMap.containsKey(topic)) {
synchronized (expiredListenersMap) {
if (!this.expiredListenersMap.containsKey(topic)) {
this.expiredListenersMap.put(topic, new ArrayList<>());
}
}
}
expiredListenersMap
.get(topic)
.add(listener);
} else {
throw new IllegalArgumentException("不存在相关主题:" + topic);
}
}
@Override
public boolean removeMessageExpiredListener(String topic, String listenerName) {
if (expiredListenersMap.containsKey(topic) && !StringUtils.isEmpty(listenerName)) {
return expiredListenersMap
.get(topic)
.removeIf(listener -> listener
.listenerName()
.equals(listenerName));
}
return false;
}
@Override
public List<MessageExpiredListener<M>> removeAllMessageExpiredListener(String topic) {
if (expiredListenersMap.containsKey(topic)) {
return expiredListenersMap.remove(topic);
}
return null;
}
/**
* <p>根据迭代器位置来使得一个元素过期,由于迭代器的弱一致性,多线程环境下可能会出现使用迭代器时
* 发生插入\删除操作,由于该消息队列对于操作严格从队尾执行,因此对于插入修改能检测到,但是由于
* 删除从队首进行,可能发生当迭代器获取下一个元素时为空,这时应该立即停止遍历,等待下一次</p>
*
* @param iterator 迭代器
* @return void
* @author Huang Yongxiang
* @date 2022/3/21 11:42
*/
public void expire(Iterator<MessageEntry<M>> iterator) throws ErrorExpiredException {
if (iterator == null) {
throw new ErrorExpiredException("过期失败,迭代器为空");
}
try {
iterator.remove();
} catch (UnsupportedOperationException e) {
throw new ErrorExpiredException("过期失败,该迭代器不支持移除操作");
} catch (IllegalStateException e) {
throw new ErrorExpiredException("过期失败,可能发生删除操作");
}
}
private void executeMessagePublishedListeners(String topic, M message) {
if (publishedListenersMap.containsKey(topic)) {
publishedListenersMap
.get(topic)
.forEach(listener -> {
try {
listener.onMessagePublished(message, messageQueues.get(topic));
} catch (Exception e) {
e.printStackTrace();
}
});
}
}
private void executeMessageExpiredListeners(String topic, M message) {
if (expiredListenersMap.containsKey(topic)) {
expiredListenersMap
.get(topic)
.forEach(listener -> {
try {
listener.onMessageExpired(message);
} catch (Exception e) {
e.printStackTrace();
}
});
}
}
@Override
public boolean publishMessageNoBlock(M message, String topic) {
if (!messageQueues.containsKey(topic)) {
log.error("发布非阻塞消息失败,所要发布到的队列主题:{}不存在", topic);
return false;
}
if (message == null) {
log.error("禁止发布空消息");
return false;
}
MessageEntry<M> messageEntry = new MessageEntry<>();
messageEntry.setMessageId(IdGenerator.getNextIdAsLong());
messageEntry.setMessage(message);
messageEntry.setExpiringTime(defaultExpiringTime > 0 ? defaultExpiringTime + System.currentTimeMillis() : -1);
if (messageQueues
.get(topic)
.publishMessage(messageEntry)) {
executeMessagePublishedListeners(topic, message);
return true;
}
return false;
}
public boolean publishMessageBlock(M message, String topic) {
if (!messageQueues.containsKey(topic)) {
log.error("发布阻塞消息失败,所要发布到的队列主题:{}不存在", topic);
return false;
}
if (message == null) {
log.error("禁止发布空消息");
return false;
}
try {
MessageEntry<M> messageEntry = new MessageEntry<>();
messageEntry.setMessageId(IdGenerator.getNextIdAsLong());
messageEntry.setMessage(message);
messageEntry.setExpiringTime(defaultExpiringTime > 0 ? defaultExpiringTime + System.currentTimeMillis() : -1);
if (messageQueues
.get(topic)
.publishMessageSync(messageEntry)) {
executeMessagePublishedListeners(topic, message);
} else {
return false;
}
} catch (InterruptedException e) {
log.warn("发布可阻塞消息发生异常,等待时被异常占用:{}", e.getMessage());
}
return false;
}
private void scheduleExpiringCheckSingleThread() {
executorService = Executors.newSingleThreadScheduledExecutor(runnable -> {
Thread thread = new Thread(runnable, "ExpiringCheckSingleThread");
thread.setDaemon(true);
return thread;
});
executorService.scheduleAtFixedRate(() -> {
for (Map.Entry<String, MessageQueue<MessageEntry<M>>> entry : messageQueues.entrySet()) {
for (Iterator<MessageEntry<M>> it = entry
.getValue()
.iterator(); it.hasNext(); ) {
MessageEntry<M> message = it.next();
//如果达到过期时间则通知其过期
if (message.expiringTime >= 0 && message.expiringTime <= System.currentTimeMillis()) {
try {
log.debug("messageId:{},消息内容:{},已过期", message.messageId, message.message);
expire(it);
executeMessageExpiredListeners(entry.getKey(), message.message);
} catch (ErrorExpiredException e) {
log.error("主题:{},消息ID:{}过期失败:{}", entry.getKey(), message.getMessageId(), e.getMessage());
}
}
}
}
}, 1, 1, TimeUnit.MILLISECONDS);
}
private void scheduleExpiringCheckTopicConcurrency() {
executorService = Executors.newSingleThreadScheduledExecutor(runnable -> {
Thread thread = new Thread(runnable, "ExpiringCheckTopicConcurrencyThread");
thread.setDaemon(true);
return thread;
});
executorService.scheduleAtFixedRate(() -> {
for (Map.Entry<String, MessageQueue<MessageEntry<M>>> entry : messageQueues.entrySet()) {
Runnable work = () -> {
try {
for (Iterator<MessageEntry<M>> it = entry
.getValue()
.iterator(); it.hasNext(); ) {
MessageEntry<M> message = it.next();
//如果达到过期时间则通知其过期
if (message.expiringTime >= 0 && message.expiringTime <= System.currentTimeMillis()) {
try {
log.debug("messageId:{}已过期", message.messageId);
expire(it);
executeMessageExpiredListeners(entry.getKey(), message.message);
} catch (ErrorExpiredException e) {
log.error("主题:{},消息ID:{}过期失败:{}", entry.getKey(), message.getMessageId(), e.getMessage());
}
}
}
} catch (Exception ignored) {
}
};
concurrencyThreads.submit(work);
}
}, 1000, 1000, TimeUnit.MILLISECONDS);
}
@Setter
@Accessors(chain = true)
public static class Builder<M> {
/**
* 默认过期时间,对所有消息设置,-1则表示消息均为永久性消息,除非消费者取出,否则将会一直存在。谨慎使用!
*/
private long defaultExpiringTime = -1;
/**
* 是否允许设置单个消息的过期时间
*/
private boolean isAllowSetEntryExpired = false;
/**
* 允许的最大主题数
*/
private int allowMaxTopicCount = 255;
/**
* 允许每个队列的最大消息数
*/
private int allowMaxMessageCountPerQueue = 1000;
/**
* 过期监听器的策略
*/
private ExpirationListenerPolicy listenerPolicy = ExpirationListenerPolicy.SINGLE_THREAD;
public Builder<M> setDefaultExpiringTime(long defaultExpiringTime, TimeUnit unit) {
if (unit == TimeUnit.MICROSECONDS) {
log.error("最小支持毫秒级");
throw new IllegalArgumentException("最小支持毫秒级");
}
this.defaultExpiringTime = unit.toMillis(defaultExpiringTime);
return this;
}
public <M1 extends M> MessageQueueContext<M1> build() {
MessageQueueContext<M1> messageQueueContext = new MessageQueueContext<>();
messageQueueContext.messageQueues = new ConcurrentHashMap<>(this.allowMaxTopicCount);
messageQueueContext.isAllowSetEntryExpired = this.isAllowSetEntryExpired;
messageQueueContext.allowMaxMessageCountPerQueue = this.allowMaxMessageCountPerQueue;
messageQueueContext.defaultExpiringTime = this.defaultExpiringTime;
messageQueueContext.allowMaxTopicCount = this.allowMaxTopicCount;
messageQueueContext.listenerPolicy = this.listenerPolicy;
if (this.listenerPolicy == ExpirationListenerPolicy.TOPIC_CONCURRENCY) {
messageQueueContext.concurrencyThreads = new ThreadPoolExecutor(3, 10, 30, TimeUnit.SECONDS, new LinkedBlockingQueue<>(10000), new NamedThreadFactory("messageQueueExpire"));
}
return messageQueueContext;
}
}
}
IMessageListener
/**
* 消息监听器抽象接口
*
* @Author Huang Yongxiang
* @Date 2022/11/04 17:04
* @Email 1158055613@qq.com
*/
public interface IMessageListener {
String listenerName();
}
MessagePublishedListener
/**
* 消息发布监听器,当有消息发布到消息队列时将会执行相关逻辑
*
* @Author Huang Yongxiang
* @Date 2022/11/02 14:52
* @Email 1158055613@qq.com
*/
public interface MessagePublishedListener<M> extends IMessageListener {
void onMessagePublished(M message, MessageQueue<MessageEntry<M>> queue);
default String listenerName() {
return "defaultMessagePublishedListener";
}
}
MessageExpiredListener
/**
* 消息过期监听器
*
* @Author Huang Yongxiang
* @Date 2022/11/02 15:18
* @Email 1158055613@qq.com
*/
public interface MessageExpiredListener<M> extends IMessageListener {
void onMessageExpired(M message);
default String listenerName() {
return "defaultMessageExpiredListener";
}
}
MessageEntry
/**
* 消息条目
*
* @author JingGe(* ^ ▽ ^ *)
* @date 2023-05-26 9:45
* @email 1158055613@qq.com
*/
public class MessageEntry<M> {
long messageId;
M message;
long expiringTime;
public MessageEntry(long messageId, M message, long expiringTime) {
this.messageId = messageId;
this.message = message;
this.expiringTime = expiringTime;
}
public MessageEntry() {
}
MessageEntry<M> setMessageId(long messageId) {
this.messageId = messageId;
return this;
}
MessageEntry<M> setMessage(M message) {
this.message = message;
return this;
}
MessageEntry<M> setExpiringTime(long expiringTime) {
this.expiringTime = expiringTime;
return this;
}
public long getMessageId() {
return messageId;
}
public M getMessage() {
return message;
}
public long getExpiringTime() {
return expiringTime;
}
}
MessageQueue
import lombok.extern.slf4j.Slf4j;
import java.util.Iterator;
import java.util.concurrent.BlockingQueue;
import java.util.concurrent.LinkedBlockingDeque;
import java.util.concurrent.LinkedBlockingQueue;
import java.util.function.Predicate;
/**
* 消息队列
*
* @author JingGe(* ^ ▽ ^ *)
* @date 2023-05-26 9:44
* @email 1158055613@qq.com
*/
@Slf4j
public class MessageQueue<M> {
final BlockingQueue<M> messageQueue;
public MessageQueue() {
messageQueue = new LinkedBlockingQueue<>();
}
public MessageQueue(int maxLength) {
if (maxLength <= 0) {
throw new IllegalArgumentException("最大容量应该为非负数");
}
messageQueue = new LinkedBlockingDeque<>(maxLength);
}
public MessageQueue(BlockingQueue<M> messageQueue) {
this.messageQueue = messageQueue;
}
public M takeMessageSync() throws InterruptedException {
return messageQueue.take();
}
public void clear() {
messageQueue.clear();
}
public M takeMessage() {
return messageQueue.poll();
}
public M readMessage() {
return messageQueue.peek();
}
public M tryReadMessage() {
return messageQueue.element();
}
public boolean removeIf(Predicate<? super M> predicate) {
if (predicate == null) {
return false;
}
return messageQueue.removeIf(predicate);
}
public int length() {
return messageQueue.size();
}
public boolean remove(M message) {
if (message == null) {
return false;
}
return messageQueue.remove(message);
}
public Iterator<M> iterator() {
return messageQueue.iterator();
}
public boolean publishMessageSync(M message) throws InterruptedException {
if (message == null) {
return false;
}
try {
messageQueue.put(message);
return true;
} catch (Exception e) {
e.printStackTrace();
throw e;
}
}
public boolean publishMessage(M message) {
if (message == null) {
return false;
}
try {
return messageQueue.offer(message);
} catch (Exception e) {
e.printStackTrace();
}
return false;
}
}