SpringAMQP消息适配器MessageListenerAdapter

MessageListenerAdapter 即消息监听适配器

1.简单使用默认方法

修改上一节 SpringAMQP 消息容器 - SimpleMessageListenerContainer 的 RabbitMQConfig 的 messageContainer 方法
@Bean //connectionFactory 也是要和最上面方法名保持一致
public SimpleMessageListenerContainer messageContainer(ConnectionFactory connectionFactory) {
SimpleMessageListenerContainer container = new SimpleMessageListenerContainer(connectionFactory);
container.setQueues(queue001(), queue002(), queue003()); //监听的队列
container.setConcurrentConsumers(1); //当前的消费者数量
container.setMaxConcurrentConsumers(5); // 最大的消费者数量
container.setDefaultRequeueRejected(false); //是否重回队列
container.setAcknowledgeMode(AcknowledgeMode.AUTO); //签收模式
container.setExposeListenerChannel(true);
container.setConsumerTagStrategy(new ConsumerTagStrategy() { //消费端的标签策略
@Override
public String createConsumerTag(String queue) {
return queue + "_" + UUID.randomUUID().toString();
}
});
// 1.1 适配器方式. 默认是有自己的方法名字的:handleMessage
MessageListenerAdapter adapter = new MessageListenerAdapter(new MessageDelegate());
container.setMessageListener(adapter);
return container;
}

public class MessageDelegate {
private static final Logger log = LoggerFactory.getLogger(MessageDelegate.class);
//这个handleMessage方法名要根据org.springframework.amqp.rabbit.listener.adapter包下的
// MessageListenerAdapter.ORIGINAL_DEFAULT_LISTENER_METHOD的默认值来确定
public void handleMessage(byte[] messageBody) {
log.info("默认方法, 消息内容:" + new String(messageBody));
}
}

这个 handleMessage 方法名要根据 org.springframework.amqp.rabbit.listener.adapter 包下的 MessageListenerAdapter.ORIGINAL_DEFAULT_LISTENER_METHOD 的默认值来确定,源码如下

SpringAMQP消息适配器MessageListenerAdapter

运行之前的测试用例 testSendMessage ,handleMessage 方法进行消息的消费

SpringAMQP消息适配器MessageListenerAdapter

2.采用自己指定一个方法的名字

将上面的 messageContainer 修改成如下的

 @Bean //connectionFactory 也是要和最上面方法名保持一致
public SimpleMessageListenerContainer messageContainer(ConnectionFactory connectionFactory) {
SimpleMessageListenerContainer container = new SimpleMessageListenerContainer(connectionFactory);
container.setQueues(queue001(), queue002(), queue003()); //监听的队列
container.setConcurrentConsumers(1); //当前的消费者数量
container.setMaxConcurrentConsumers(5); // 最大的消费者数量
container.setDefaultRequeueRejected(false); //是否重回队列
container.setAcknowledgeMode(AcknowledgeMode.AUTO); //签收模式
container.setExposeListenerChannel(true);
container.setConsumerTagStrategy(new ConsumerTagStrategy() { //消费端的标签策略
@Override
public String createConsumerTag(String queue) {
return queue + "_" + UUID.randomUUID().toString();
}
});
//1.2 适配器方式. 可以自己指定一个方法的名字: consumeMessage
MessageListenerAdapter adapter = new MessageListenerAdapter(new MessageDelegate());
adapter.setDefaultListenerMethod("consumeMessage");
container.setMessageListener(adapter);
return container;
}

MessageDelegate 里面的消费方法改成 consumeMessage

SpringAMQP消息适配器MessageListenerAdapter

 public void consumeMessage(byte[] messageBody) {
log.info("字节数组方法, 消息内容:" + new String(messageBody));
}

继续运行 testSendMessage, 查看到消费

SpringAMQP消息适配器MessageListenerAdapter

3. 添加一个转换器,从字节数组转换为 String

//1.3 适配器方式.也可以添加一个转换器: 从字节数组转换为String
public class TextMessageConverter implements MessageConverter {
@Override
public Message toMessage(Object object, MessageProperties messageProperties) throws MessageConversionException {

return new Message(object.toString().getBytes(), messageProperties);
}
@Override
public Object fromMessage(Message message) throws MessageConversionException {
String contentType = message.getMessageProperties().getContentType();
if(null != contentType && contentType.contains("text")) {
return new String(message.getBody());
}
return message.getBody();
}
}

toMessage 就是 Java 对象转换为 Message,fromMessage 就是 Message 转为 Java 对象

将上面的 messageContainer 修改成如下的

 @Bean //connectionFactory 也是要和最上面方法名保持一致
public SimpleMessageListenerContainer messageContainer(ConnectionFactory connectionFactory) {
SimpleMessageListenerContainer container = new SimpleMessageListenerContainer(connectionFactory);
container.setQueues(queue001(), queue002(), queue003()); //监听的队列
container.setConcurrentConsumers(1); //当前的消费者数量
container.setMaxConcurrentConsumers(5); // 最大的消费者数量
container.setDefaultRequeueRejected(false); //是否重回队列
container.setAcknowledgeMode(AcknowledgeMode.AUTO); //签收模式
container.setExposeListenerChannel(true);
container.setConsumerTagStrategy(new ConsumerTagStrategy() { //消费端的标签策略
@Override
public String createConsumerTag(String queue) {
return queue + "_" + UUID.randomUUID().toString();
}
});
//1.3 适配器方式.也可以添加一个转换器: 从字节数组转换为String
MessageListenerAdapter adapter = new MessageListenerAdapter(new MessageDelegate());
adapter.setDefaultListenerMethod("consumeMessage");
adapter.setMessageConverter(new TextMessageConverter());
container.setMessageListener(adapter);
return container;
}

关键点,这里使用的不再是字节数组了!!

SpringAMQP消息适配器MessageListenerAdapter

 //1.3 适配器方式.也可以添加一个转换器: 从字节数组转换为String
public void consumeMessage(String messageBody) {
log.info("字符串方法, 消息内容:" + messageBody);
}

写个单元测试用例,注意 contentType 要包含 text !!

 @Test
public void testSendMessage4Text() throws Exception {
//1 创建消息

MessageProperties messageProperties = new MessageProperties();
messageProperties.setContentType("text/plain");
Message message = new Message("mq 消息1234".getBytes(), messageProperties);
rabbitTemplate.send("topic001", "spring.abc", message);
}

运行单元测试

SpringAMQP消息适配器MessageListenerAdapter

4. 队列名称 和 方法名称 也可以进行一一的匹配

将上面的 messageContainer 修改成如下的

 @Bean //connectionFactory 也是要和最上面方法名保持一致
public SimpleMessageListenerContainer messageContainer(ConnectionFactory connectionFactory) {
SimpleMessageListenerContainer container = new SimpleMessageListenerContainer(connectionFactory);

container.setQueues(queue001(), queue002(), queue003()); //监听的队列
container.setConcurrentConsumers(1); //当前的消费者数量
container.setMaxConcurrentConsumers(5); // 最大的消费者数量
container.setDefaultRequeueRejected(false); //是否重回队列
container.setAcknowledgeMode(AcknowledgeMode.AUTO); //签收模式
container.setExposeListenerChannel(true);
container.setConsumerTagStrategy(new ConsumerTagStrategy() { //消费端的标签策略
@Override
public String createConsumerTag(String queue) {
return queue + "_" + UUID.randomUUID().toString();
}
});
// 2 适配器方式: 我们的队列名称 和 方法名称 也可以进行一一的匹配
MessageListenerAdapter adapter = new MessageListenerAdapter(new MessageDelegate());
adapter.setMessageConverter(new TextMessageConverter());
Map<string> queueOrTagToMethodName = new HashMap<>();
queueOrTagToMethodName.put("queue001", "method1");
queueOrTagToMethodName.put("queue002", "method2");
adapter.setQueueOrTagToMethodName(queueOrTagToMethodName);
container.setMessageListener(adapter);
return container;
}
/<string>
SpringAMQP消息适配器MessageListenerAdapter

 // 2 适配器方式: 我们的队列名称 和 方法名称 也可以进行一一的匹配
public void method1(String messageBody) {
log.info("method1 收到消息内容:" + new String(messageBody));
}
public void method2(String messageBody) {
log.info("method2 收到消息内容:" + new String(messageBody));
}

看一下之前建立的绑定关系

SpringAMQP消息适配器MessageListenerAdapter

修改一下单元测试用例 @Test
public void testSendMessage4Text() throws Exception {

//1 创建消息
MessageProperties messageProperties = new MessageProperties();
messageProperties.setContentType("text/plain");
Message message = new Message("mq 消息1234".getBytes(), messageProperties);
rabbitTemplate.send("topic001", "spring.abc", message);
rabbitTemplate.send("topic002", "rabbit.abc", message);
}

运行测试, 查看到两个队列的消费

SpringAMQP消息适配器MessageListenerAdapter

综上,通过上面 MessageListenerAdapter 的使用代码,我们可以看出它有如下核心属性

  • defaultListenerMethod 默认监听方法名称:用于设置监听方法的名称
  • delegate 委派对象: 实际真实的委派对象,用于处理消息
  • queueOrTagMethodName 队列标识于方法名称组成的集合。
  • 可以一一进行队列于方法名称的匹配。
  • 队列和方法名称绑定,即指定队列里的消息会被绑定的方法所接受处理。

关注我:私信回复“555”获取往期Java高级架构资料、源码、笔记、视频Dubbo、Redis、Netty、zookeeper、Spring cloud、分布式、高并发等架构技术往期架构视频截图


分享到:


相關文章: