spring boot集成ActiveMQ

Posted 不去天涯

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了spring boot集成ActiveMQ相关的知识,希望对你有一定的参考价值。

从官方demo开始

spring-cloud其他文章:
我们有必要花时间了解spring boot吗?
spring boot自动化配置原理
spring-cloud中eureka进行服务治理

spring boot提供了对JMS系统的支持,当前,只默认提供了ActiveMQ和ActiveMQ Artemis的自动配置支持。

我们一起看下spring boot如何集成ActiveMQ。首先从模仿官方的demo开始。

新建一个spring boot工程

新建方法参见我们有必要花时间了解spring boot吗?的最后一节,如何给使用工具创建spring boot项目。

选择Lombok、JMS、Web三个starter组件,点击完成。Lombok用来给模型自动添加setter、getter、constructor方法。

工具帮我们自动创建好了工程结构,添加进来了刚才选中的starter依赖。对于spring boot工程的结构,可以到官方网站了解一下,了解了之后才能明白工具到底做了什么,这样,没有工具也能创建项目,才不会编程工具的奴隶。

修改ActiveMQ配置

在application.properties中添加如下配置(spring boot的配置项默认都在这个文件里,也可以自己添加新的文件,实现不同配置的分开存储):

spring.activemq.in-memory=true
spring.activemq.pool.enabled=false

备注:实际上这两个是默认值,不配置情况下即是如此。如果你是使用spring tool suit工具或者集成到了eclipse,那么当你在application.properties里输入spring.activemq.in-memory的时候,就会提示给你默认值是什么:

添加Producer类

@RestController
public class Producer 
    @Autowired
    private JmsMessagingTemplate jmsMessagingTemplate;

    @Autowired
    private Queue queue;

    @RequestMapping("/sendMsg")
    public void send(String msg) 
        this.jmsMessagingTemplate.convertAndSend(this.queue, msg);
    

这里做一个小的修改,使用熟悉的rest服务方式来触发发送消息的方法。

@RestController相当于指定类内部的@RequestMapping返回值都是json,就不需要添加@ResponseBody注解了。

这里的jmsMessagingTemplate和queue都是用@Autowired注解自动注入的。注意这里的jmsMessagingTemplate,在整个工程里都没有配置实例化的地方。

其实,jmsMessagingTemplate的实例化是spring boot的autoconfigure自动注入的。

实现可以参见JmsAutoConfiguration.java的代码。大概意思就是当我们引入了jms相关的包,比如activemq的包,又定义了ConnectionFactory的实例,那么他就会自动生成一个JmsTemplate实例。

ConnectionFactory在哪里自动创建的呢?看这个ActiveMQConnectionFactoryConfiguration.java。大概意思是,如果没有ConnectionFactory实例,就自动创建一个实例。

所以说,只要引入了spring-boot-starter-activemq那么就会给我们自动创建一个JmsTemplate,相关的连接配置从application.properties,如果里面没有配置的话就会使用ActiveMQProperties.java的默认值。

添加Consumer类

@Component
public class Consumer 

    @JmsListener(destination = "sample.queue")
    public void receiveQueue(String text) 
        System.out.println(text);
    

JmsListener是spring-jms提供的一个注解,会实例化一个Jms的消息监听实例,也就是一个异步的消费者。

添加JMS的注解扫描

@SpringBootApplication
@EnableJms
public class ActivemqDemoApplication 
    @Bean
    public Queue queue() 
        return new ActiveMQQueue("sample.queue");
    

    public static void main(String[] args) 
        SpringApplication.run(ActivemqDemoApplication.class, args);
    

@EnableJms会启动jms的注解扫描,相当于<jms:annotation-d riven/>

启动项目

在ActivemqDemoApplication启动java工程。

然后在浏览器输入:http://localhost:8080/sendMsg?msg=HelloActiveMQ

就可以看到eclipse的控制台输出了对应的消息。

连接外部的ActiveMQ

修改配置

官方的实例用的是in-memory的ActiveMQ。然而我们实际使用的时候都是连接外部共用的ActiveMQ服务。所以,我们修改一下配置,来使用外部的ActiveMQ服务。

spring.activemq.broker-url=tcp://localhost:61616
#spring.activemq.broker-url=failover:(tcp://localhost:61616,tcp://localhost:61617)
spring.activemq.close-timeout=5000
spring.activemq.in-memory=false
#spring.activemq.pool.enabled=true
spring.activemq.pool.max-connections=100
spring.activemq.send-timeout=3000

这里被注释掉的spring.activemq.broker-url是对应ActiveMQ集群时候的broker-url配置。

spring.activemq.pool.enabled备注释掉了,是因为作者用的spring boot 1.5.6对应的jms-starter没有包含activemq-pool的依赖引入,所以不能设置pool.enabled=true,如果要使用需要自己添加activemq-pool的依赖包:

<dependency>
   <groupId>org.apache.activemq</groupId>
    <artifactId>activemq-pool</artifactId>
   <!--  <version>5.7.0</version> -->
</dependency>

启动服务

当我们没有启动ActiveMQ的时候,spring boot启动失败了:

***************************
APPLICATION FAILED TO START
***************************

Description:

Field jmsMessagingTemplate in com.example.demo.activemq.Producer required a bean of type 'org.springframework.jms.core.JmsMessagingTemplate' that could not be found.
    - Bean method 'jmsMessagingTemplate' not loaded because Ancestor org.springframework.boot.autoconfigure.jms.JmsAutoConfiguration did not match


Action:

Consider revisiting the conditions above or defining a bean of type 'org.springframework.jms.core.JmsMessagingTemplate' in your configuration.

但是失败的消息提示很诡异,告诉的是找不到JmsMessagingTemplate,而不是提示连接不上ActiveMQ。。。

按照提示信息里面说的设置debug=true,能够打印出auto-configure中相关类的匹配信息,但是仍然不能直接定位到是ActiveMQ服务连接不上。

如果遇到上述问题,可以尝试检查ActiveMQ服务是否能够连接。

一个jmsTemplate支持queue和topic

作者使用的版本已经支持一个jmsTemplate支持queue和topic的消息发送,只需要指定不同的destination即可。

但是消息的接收,也就是@JmsListener如果不指定独立的containerFactory的话是只能消费queue消息的,如果要能够同时接收topic消息,需要给topic对应的@JmsListener增加containerFactory配置:

    //需要给topic定义独立的JmsListenerContainer
    @Bean
    public JmsListenerContainerFactory<?> jmsListenerContainerTopic(ConnectionFactory activeMQConnectionFactory) 
        DefaultJmsListenerContainerFactory bean = new DefaultJmsListenerContainerFactory();
        bean.setPubSubDomain(true);
        bean.setConnectionFactory(activeMQConnectionFactory);
        return bean;
    


    @JmsListener(destination = "sample.topic", containerFactory="jmsListenerContainerTopic")
    public void receiveTopic(String text) 
        System.out.println(text);
    

本文的项目代码已经上传到spring-boot-activemq-demo

以上是关于spring boot集成ActiveMQ的主要内容,如果未能解决你的问题,请参考以下文章

spring boot集成ActiveMQ

Spring boot 集成ActiveMQ(包含双向队列实现)

Spring Boot(XML配置)和Jasypt集成

activeMQ学习(与springboot集成)

activemq 学习系列 activemq 与 spring boot 整合

Camel ActiveMQ + Spring boot 不读取 spring activemq 配置