事先说明,本博客关于ActiveMQ的文章都是基于ActiveMQ5.10版本。
初步用过ActiveMQ但又没去研究过源码的朋友肯定有些好奇ActiveMQ中消费者是如何接收消息的呢?本文我就和大家一起从源码角度来初步探讨消费者接收消息的过程。
我们知道,消息传送有两种模型:点对点(P2P)和发布订阅(PUB/SUB),队列模式中,消息生产者叫做发送者,消息消费者叫做接收者,而在发布订阅模式中,消息生产者叫发布者,消息消费者叫订阅者。点对点模型中队列(Queue)是消息发送和接收的途径和通道,他保证了一个消息最多只能被一个消费者消费,而发布订阅模型中,消息发送和接收的途径是主题(Topic),所有订阅主题的消费者,都可以接收到该主题发布的消息,所以在这个模型中,消息可以被多个消费者消费。
1)我们先来看看在点对点模型中消费者是如何接收消息的
如果直接使用过ActiveMQ API的朋友,一定知道消息接收者可以通过两种方式接收消息,一种是使用同步效果的MessageConsumer#receive() 和异步的使用消息监听器的MessageConsumer#setMessageListener(MessageListener listener) 。值得注意的是,在同一个org.apache.activemq.ActiveMQSession会话对象下面的消费者,如果有的是采用消息监听器接收消息,则那些采用同步receive() 接收消息的消费者会抛出 IllegalStateException("Cannot synchronously receive a message when a MessageListener is set")异常,也就是说,同一个Session下面,要么消费者都使用消息监听器,要么都使用receive() 同步接收。
这是为什么呢?我们先看下org.apache.activemq.ActiveMQMessageConsumer同步接收的源代码:
@Override
public Message receive() throws JMSException {
checkClosed();
checkMessageListener();
sendPullCommand(0); // 如果预取数为0,则主动向JMS服务器发送拉取消息的报文
MessageDispatch md = dequeue(-1);
if (md == null) {
return null;
}
beforeMessageIsConsumed(md);
afterMessageIsConsumed(md, false); // 给JMS服务器发送接收消息的应答报文
return createActiveMQMessage(md); // 取出消息副本并返回
}
上面的checkMessageListener()就是去做检查的,请看:
protected void checkMessageListener() throws JMSException {
// 去调用所属会话的checkMessageListener();方法