ActiveMQ 持久化配置

ActiveMQ 持久化配置

修改配置文件activemq.xml

修改persistenceAdapter

将下面这段配置,注释掉

<persistenceAdapter>
    <kahaDB directory="${activemq.data}/kahadb"/>
</persistenceAdapter>

然后更新成下面的配置:

<persistenceAdapter>
    <jdbcPersistenceAdapter dataSource="#mysql-ds"/> 
</persistenceAdapter>
定义数据源bean

在broker节点的外面,定义id是mysql-ds的bean,如下:

<bean id="mysql-ds" class="org.apache.commons.dbcp2.BasicDataSource" destroy-method="close">
	<property name="driverClassName" value="com.mysql.jdbc.Driver"/> 
	<property name="url" value="jdbc:mysql://localhost:3306/activemq?relaxAutoCommit=true"/>
	<property name="username" value="root"/>
	<property name="password" value="root"/> 
	<property name="poolPreparedStatements" value="true"/> 
</bean> 

创建数据

在mysql中创建名称为activemq的数据库,并设置编码方式是UTF-8。

重启ActiveMq

修改代码

在生产者上设置持久化模式sender.setDeliveryMode(DeliveryMode.PERSISTENT);,完整代码如下:

import javax.jms.DeliveryMode;
import javax.jms.MapMessage;
import javax.jms.Queue;
import javax.jms.QueueConnection;
import javax.jms.QueueConnectionFactory;
import javax.jms.QueueSession;
import javax.jms.Session;
import org.apache.activemq.ActiveMQConnection;
import org.apache.activemq.ActiveMQConnectionFactory;

/**
 * Queue 方式消息发送者
 */
public class QueueSender {

	// 发送次数
	public static final int SEND_NUM = 5;
	// tcp 地址
	public static final String BROKER_URL = "tcp://localhost:61616";
	// 目标,在ActiveMQ管理员控制台创建 http://localhost:8161/admin/queues.jsp
	public static final String DESTINATION = "hoo.mq.persistence";

	/**
	 * <b>function:</b> 发送消息
	 * 
	 * @author hoojo
	 * @createDate 2013-6-19 下午12:05:42
	 * @param session
	 * @param sender
	 * @throws Exception
	 */
	public static void sendMessage(QueueSession session, javax.jms.QueueSender sender) throws Exception {
		for (int i = 0; i < SEND_NUM; i++) {
			String message = "发送消息第" + (i + 1) + "条";

			MapMessage map = session.createMapMessage();
			map.setString("text", message);
			map.setLong("time", System.currentTimeMillis());
			System.out.println(map);

			sender.send(map);
		}
	}

	public static void run() throws Exception {

		QueueConnection connection = null;
		QueueSession session = null;
		try {
			// 创建链接工厂
			QueueConnectionFactory factory = new ActiveMQConnectionFactory(ActiveMQConnection.DEFAULT_USER,
					ActiveMQConnection.DEFAULT_PASSWORD, BROKER_URL);
			// 通过工厂创建一个连接
			connection = factory.createQueueConnection();
			// 启动连接
			connection.start();
			// 创建一个session会话
			session = connection.createQueueSession(Boolean.TRUE, Session.AUTO_ACKNOWLEDGE);
			// 创建一个消息队列
			Queue queue = session.createQueue(DESTINATION);
			// 创建消息发送者
			javax.jms.QueueSender sender = session.createSender(queue);
			// 设置持久化模式
			sender.setDeliveryMode(DeliveryMode.PERSISTENT);
			sendMessage(session, sender);
			// 提交会话
			session.commit();

		} catch (Exception e) {
			throw e;
		} finally {
			// 关闭释放资源
			if (session != null) {
				session.close();
			}
			if (connection != null) {
				connection.close();
			}
		}
	}

	public static void main(String[] args) throws Exception {
		QueueSender.run();
	}
}

消费者的代码不用修改:

import javax.jms.JMSException;
import javax.jms.MapMessage;
import javax.jms.Message;
import javax.jms.MessageListener;
import javax.jms.Queue;
import javax.jms.QueueConnection;
import javax.jms.QueueConnectionFactory;
import javax.jms.QueueSession;
import javax.jms.Session;
import org.apache.activemq.ActiveMQConnection;
import org.apache.activemq.ActiveMQConnectionFactory;

/**
 * 消息接收者
 */
public class QueueReceiver {

	// tcp 地址
	public static final String BROKER_URL = "tcp://localhost:61616";
	// 目标,在ActiveMQ管理员控制台创建 http://localhost:8161/admin/queues.jsp
	public static final String TARGET = "hoo.mq.persistence";

	public static void run() throws Exception {

		QueueConnection connection = null;
		QueueSession session = null;
		try {
			// 创建链接工厂
			QueueConnectionFactory factory = new ActiveMQConnectionFactory(ActiveMQConnection.DEFAULT_USER,
					ActiveMQConnection.DEFAULT_PASSWORD, BROKER_URL);
			// 通过工厂创建一个连接
			connection = factory.createQueueConnection();
			// 启动连接
			connection.start();
			// 创建一个session会话
			session = connection.createQueueSession(Boolean.TRUE, Session.AUTO_ACKNOWLEDGE);
			// 创建一个消息队列
			Queue queue = session.createQueue(TARGET);
			// 创建消息制作者
			javax.jms.QueueReceiver receiver = session.createReceiver(queue);

			receiver.setMessageListener(new MessageListener() {
				public void onMessage(Message msg) {
					if (msg != null) {
						MapMessage map = (MapMessage) msg;
						try {
							System.out.println(map.getLong("time") + "接收#" + map.getString("text"));
						} catch (JMSException e) {
							e.printStackTrace();
						}
					}
				}
			});
			// 休眠100ms再关闭
			Thread.sleep(1000 * 3);

			// 提交会话,
			session.commit();

		} catch (Exception e) {
			throw e;
		} finally {
			// 关闭释放资源
			if (session != null) {
				session.close();
			}
			if (connection != null) {
				connection.close();
			}
		}
	}

	public static void main(String[] args) throws Exception {
		QueueReceiver.run();
	}
}

测试

ActiveMQ持久化 就是由生产者产生的数据,在没有被消费者消费时,会先保到数据库中,当数据被消费者消费后,会从数据库中删除。

基本以上的原理,测试时,从运行生产者的程序,然后在数据库中看到ActiveMQ自动生成了三张表:activemq_acksactivemq_lockactivemq_msgs,在 activemq_msgs表中看到了生产的数据。
然后再运行消费者端的程序,执行完成后,activemq_msgs表中的数据消失。

持久化配置的报错

java.lang.ClassNotFoundException: org.apache.commons.dbcp.BasicDataSource

将 ActiveMQ 持久化改成MySQL配置后,启动时报错了。主要就是找不到org.apache.commons.dbcp.BasicDataSource 这个类。
经过网上查找资料,BasicDataSource 位于commons-dbcp2-xx.jar,(位置在ActiveMQ的lib/optional包中有 );
报错原因是ActiveMQ配置MySQL数据源时,要把class写作org.apache.commons.dbcp2.BasicDataSource。
配置如下:

<bean id="mysql-ds" class="org.apache.commons.dbcp2.BasicDataSource" destroy-method="close">
	<property name="driverClassName" value="com.mysql.jdbc.Driver"/> 
	<property name="url" value="jdbc:mysql://localhost:3306/activemq?relaxAutoCommit=true"/>
	<property name="username" value="root"/>
	<property name="password" value="root"/> 
	<property name="poolPreparedStatements" value="true"/> 
</bean> 
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值