activemq 事务--遇到异常始终回滚

来源:互联网 发布:网络鲜花店 上海 编辑:程序博客网 时间:2024/06/09 17:24

activemq 事务--遇到异常始终回滚,


1. 启用消息事务
<property name="sessionTransacted" value="true"/>




2. 当消息在消费的时候,如果用户程序抛出Exception,则消息会回滚重传(mq里面的未消费消息数目不变), 
spring 的activemq默认最多重传6次, 超过6次,即使抛出了异常,这个消费仍然被消费不可回滚。
jms.redeliveryPolicy.maximumRedeliveries=6(默认)
我们可以设置brokerURL里面的jms.redeliveryPolicy.maximumRedeliveries=-1
-1表示可以无限次重传,0表示不重传。


注意中间要用 html转移符 &amp;     (就是url查询字符串里的&字符)

3.消息队列回滚的触发--程序本身运行异常,比如数据库操作异常,如果自己用程序去检查数据库更新条数,如果小于1,程序可以人为编写代码,人为

抛出一个RuntimeException. 这两种情况都能触发消息的rollback

如在onMesssage()里,或者内部嵌套的方法里

throw new RuntimeException("Update DB failed.");



最终配置文件如下

appContext.xml

<?xml version="1.0" encoding="UTF-8"?><beans xmlns="http://www.springframework.org/schema/beans"xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:context="http://www.springframework.org/schema/context"xsi:schemaLocation="  http://www.springframework.org/schema/beans        http://www.springframework.org/schema/beans/spring-beans-3.0.xsd  http://www.springframework.org/schema/context                http://www.springframework.org/schema/context/spring-context-3.0.xsd"><import resource="mybatisContext.xml" /><import resource="shardingContext.xml" /><!-- 配置JMS连接工厂  --><!-- 测试环境 --><bean id="jmsFactory" class="org.apache.activemq.spring.ActiveMQConnectionFactory"><property name="brokerURL"value="failover:(tcp://10.0.0.9:61616)?randomize=false&timeout=3000&initialReconnectDelay=100&jms.useAsyncSend=true&jms.redeliveryPolicy.maximumRedeliveries=-1" /></bean><!-- 配置JMS模版 --><bean id="jmsTemplate" class="org.springframework.jms.core.JmsTemplate"><property name="connectionFactory"><!-- lets wrap in a pool to avoid creating a connection per send --><bean class="org.springframework.jms.connection.SingleConnectionFactory"><property name="targetConnectionFactory"><ref local="jmsFactory" /></property></bean></property></bean><!-- MQ监听者 --><bean id="smsSendForMQListener"class="org.springframework.jms.listener.DefaultMessageListenerContainer"><property name="connectionFactory"><ref bean="jmsFactory" /></property><property name="destinationName"><value>DistributePics</value></property><property name="messageListener"><ref bean="saveListener" /></property><!-- mq事务控制,如果抛异常了就回滚消息 --><property name="sessionTransacted" value="true"/></bean><!-- Spring MVC --><context:component-scan base-package="com.x.service"></context:component-scan><context:component-scan base-package="com.x.controller.**"></context:component-scan><context:component-scan base-package="com.x.listener"></context:component-scan><context:component-scan base-package="com.x.spring"></context:component-scan></beans>


java程序如下:


package com.x.imgapp.listener;import java.util.concurrent.Future;import javax.jms.JMSException;import javax.jms.Message;import javax.jms.MessageListener;import javax.jms.ObjectMessage;import javax.jms.BytesMessage;import org.springframework.stereotype.Component;import com.x.imgapp.common.model.MsgBean;import com.x.imgapp.concurrent.AppExecutor;import com.x.imgapp.concurrent.MoveThread;import org.apache.activemq.command.ActiveMQBytesMessage;import org.apache.activemq.util.ByteSequence;import org.slf4j.Logger;import org.slf4j.LoggerFactory;/** *  * @author frank.liu * */@Component("saveListener")public class SaveDBListener implements MessageListener {private static final Logger logger = LoggerFactory.getLogger(SaveDBListener.class);private static Long receiveCount = 0L;@Overridepublic void onMessage(Message message) {logger.info("Step into onMessage()------------------------------->");ActiveMQBytesMessage msg = (ActiveMQBytesMessage) message;ByteSequence sequence = msg.getContent();String msgStr = new String(sequence.data);////Future<Integer> future = AppExecutor.getExecutor().submit(new MoveThread(msgStr));logger.info("receiveCount:{}", ++receiveCount);throw new RuntimeException("Update DB failed.");}}