spring结合activemq使用

依赖如下:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
<dependency>
    <groupId>${spring.groupId}</groupId>
    <artifactId>spring-jms</artifactId>
    <version>${spring.version}</version>
</dependency>
<dependency>
    <groupId>org.apache.activemq</groupId>
    <artifactId>activemq-all</artifactId>
    <version>5.9.0</version>
</dependency>
<dependency>
    <groupId>org.apache.activemq</groupId>
    <artifactId>activemq-pool</artifactId>
    <version>5.9.0</version>
</dependency>
<dependency>
    <groupId>xml-apis</groupId>
    <artifactId>xml-apis</artifactId>
    <version>1.4.01</version>
</dependency>

发送和接收队列消息的配置:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
 <bean id="jmsFactory" class="org.apache.activemq.pool.PooledConnectionFactory" destroy-method="stop">
    <property name="connectionFactory">
        <bean class="org.apache.activemq.ActiveMQConnectionFactory">
            <property name="brokerURL">
                <value>tcp://192.168.10.65:61616</value>
            </property>
        </bean>
    </property>
    <property name="maxConnections" value="100"></property>
</bean>
<bean id="jmsTemplate" class="org.springframework.jms.core.JmsTemplate">
    <property name="connectionFactory" ref="jmsFactory"></property>
    <!--如果使用jmsTemplate发送时不指定destination,则发送到默认的destination,即这里配置的-->
    <property name="defaultDestination" ref="destination"></property>
    <property name="messageConverter">
        <bean class="org.springframework.jms.support.converter.SimpleMessageConverter"></bean>
    </property>
</bean>
<bean id="destination" class="org.apache.activemq.command.ActiveMQQueue">
    <constructor-arg index="0" value="spring-queue"></constructor-arg>
</bean>

发送和接收消息都是通过JmsTemplate类操作的。在JmsTemplate中有defaultDestination属性,表示默认的目的地,可以是队列也可以是Topic。当你配置了默认的destination后,在使用jmsTemplate发送消息不指定destination时就使用默认的destination。

发送消息的代码:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;
import org.springframework.jms.core.JmsTemplate;
import org.springframework.jms.core.MessageCreator;
import org.springframework.stereotype.Component;
import javax.jms.JMSException;
import javax.jms.Message;
import javax.jms.Session;
import javax.jms.TextMessage;
/**
 * @author j.tommy
 * @date 2017-04-06 17:59.
 */
@Component
public class QueueSendTest {
    @Autowired
    private JmsTemplate jmsTemplate;
    public static void main(String[] args) {
        ApplicationContext ac = new ClassPathXmlApplicationContext("application*.xml");
        QueueSendTest qst = ac.getBean(QueueSendTest.class);
        qst.jmsTemplate.send(new MessageCreator() {
            @Override
            public Message createMessage(Session session) throws JMSException {
                TextMessage msg = session.createTextMessage("Spring msg ===");
                return msg;
            }
        });
    }
}

接收消息的代码为:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;
import org.springframework.jms.core.JmsTemplate;
import org.springframework.stereotype.Component;

/**
 * @author j.tommy
 * @date 2017-04-06 18:02.
 */
@Component
public class QueueRecvTest {
    @Autowired
    private JmsTemplate jmsTemplate;
    public static void main(String[] args) {
        ApplicationContext ac = new ClassPathXmlApplicationContext("application*.xml");
        QueueRecvTest qrt = ac.getBean(QueueRecvTest.class);
        String msg = (String) qrt.jmsTemplate.receiveAndConvert();
        System.out.println("recv msg : " + msg);
    }
}

先启动发送端,再启动接收端就可以接收到消息了。

发送和接收topic消息

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
<bean id="jmsFactory" class="org.apache.activemq.pool.PooledConnectionFactory" destroy-method="stop">
    <property name="connectionFactory">
        <bean class="org.apache.activemq.ActiveMQConnectionFactory">
            <property name="brokerURL">
                <value>tcp://192.168.10.65:61616</value>
            </property>
        </bean>
    </property>
    <property name="maxConnections" value="100"></property>
</bean>
<bean id="jmsTemplate" class="org.springframework.jms.core.JmsTemplate">
    <property name="connectionFactory" ref="jmsFactory"></property>
    <!--如果使用jmsTemplate发送时不指定destination,则发送到默认的destination,即这里配置的-->
    <property name="defaultDestination" ref="destinationTopic"></property>
    <property name="messageConverter">
        <bean class="org.springframework.jms.support.converter.SimpleMessageConverter"></bean>
    </property>
</bean>
<bean id="destinationTopic" class="org.apache.activemq.command.ActiveMQTopic">
    <constructor-arg index="0" value="spring-topic"></constructor-arg>
</bean>
<bean id="jmsContainer" class="org.springframework.jms.listener.DefaultMessageListenerContainer">
    <property name="connectionFactory" ref="jmsFactory"></property>
    <property name="destination" ref="destinationTopic"></property>
    <property name="messageListener" ref="messageListener"></property>
</bean>
<bean id="messageListener" class="com.hhly.jms.MyMessageListener"></bean>

发送端的代码与队列一样。
接收端:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
import javax.jms.JMSException;
import javax.jms.Message;
import javax.jms.MessageListener;
import javax.jms.TextMessage;
/**
 * @author j.tommy
 * @date 2017-04-07 09:25.
 */
public class MyMessageListener implements MessageListener {
    @Override
    public void onMessage(Message message) {
        if (message instanceof TextMessage) {
            TextMessage tm = (TextMessage) message;
            try {
                System.out.println("接收到消息:" + tm.getText());
            } catch (JMSException e) {
                e.printStackTrace();
            }
        }
    }
}

启动发送端就可以了。

说明:

  1. 在JmsTemplate中配置了defaultDestination属性后,在发送时如果不指定destination将使用默认的。建议在发送时指定Destionation。
    如图:
  2. 最佳实践
  3. 最好不要使用failover,使用failover在broker异常时会阻塞住。使用spring的DefaultMessageListenerContainer会自动重连,如图:
    。failover能够在你配置了多个activemq时自动选择可用的broker,但broker异常会导致线程阻塞。如果一定要使用failover,比如就配置了一个activemq,建议数据放到队列,同时队列设置一定的容量,有专门的线程从队列取数据放入mq。
  4. log4j日志级别不能配置error,否则spring和activemq的日志显示不了。