Apache ActiveMQ - 3 - ActiveMQ消息中间件

消息中间件


消息中间件(MOM:Message Orient middleware)


用途和优点:

1. 将数据从一个应用程序传送到另一个应用程序,或者从软件的模块传送到另一个模块。

2. 负责建立网络通信的通道,进行数据的可靠传送。

3. 保证数据不重发,不丢失

4. 能够实现跨平台操作,能够为不同操作系统上的软件集成技工数据传送服务。



MQ :

MessageQueue,消息队列。

作用 : 就是一个消息的接受和转发的容器,可用于消息推送。就是一个消息的接受和转发的容器,可用于消息推送。



ActiveMQ特性列表

1. 多种语言和协议编写客户端。语言: Java, C, C++, C#, Ruby, Perl, Python, PHP。应用协议: OpenWire,Stomp REST,WS Notification,XMPP,AMQP

2. 完全支持JMS1.1和J2EE 1.4规范 (持久化,XA消息,事务)

3. 对Spring的支持,ActiveMQ可以很容易内嵌到使用Spring的系统里面去,而且也支持Spring2.0的特性

4. 通过了常见J2EE服务器(如 Geronimo,JBoss 4, GlassFish,WebLogic)的测试,其中通过JCA 1.5 resource adaptors的配置,可以让ActiveMQ可以自动的部署到任何兼容J2EE 1.4 商业服务器上

5. 支持多种传送协议:in-VM,TCP,SSL,NIO,UDP,JGroups,JXTA

6. 支持通过JDBC和journal提供高速的消息持久化

7. 从设计上保证了高性能的集群,客户端-服务器,点对点

8. 支持Ajax

9. 支持与Axis的整合

10. 可以很容易得调用内嵌JMS provider,进行测试



什么情况下使用ActiveMQ?

1. 多个项目之间集成

          跨平台

          多语言

          多项目

2. 降低系统间模块的耦合度,解耦

          软件扩展性

3. 系统前后端隔离

          前后端隔离,屏蔽高安全区



ActiveMQ接收和发送基本流程


发送消息的基本步骤:

1. 创建连接使用的工厂类JMS ConnectionFactory

2. 使用管理对象JMS ConnectionFactory建立连接Connection,并启动

3. 使用连接Connection建立会话Session

4. 使用会话Session和管理对象Destination创建消息生产者MessageSender

5. 使用消息生产者MessageSender发送消息.


消息接受者从JMS接收消息的步骤:

1. 创建连接使用的工厂类JMS ConnectionFactory

2. 使用管理对象JMS ConnectionFactory建立连接Connection,并启动

3. 使用连接Connection建立会话Session

4. 使用会话Session和管理独享Destination创建消息接受者MessageReceiver

5. 使用消息接受者MessageReceiver接收消息,需要用SetMessageListener将MessageListener接口绑定到MessageReceiver,消息接收者必须实现了MessageListener接口,需要定义onMessage事件方法.



Demo

下面来一个Demo,当然了,前提是你的MQ已经安装好了。

package com.activemq.activemq1_queue;

import javax.jms.DeliveryMode;
import javax.jms.MapMessage;
import javax.jms.Queue;
import javax.jms.QueueConnection;
import javax.jms.QueueConnectionFactory;
import javax.jms.QueueSender;
import javax.jms.QueueSession;
import javax.jms.Session;

import org.apache.activemq.ActiveMQConnection;
import org.apache.activemq.ActiveMQConnectionFactory;
import org.apache.activemq.ActiveMQConnectionMetaData;

/**
* Queue队列方式发送点对点消息数据
* 
* Queue方式消息发送
* 
* @author CYX
* @time 2016年12月19日下午7:29:07
*/
public class QueueSenderMeesage {

	// 消息发送次数
	public static final int SEND_SEND = 5;
	// tcp地址
	public static final String BROKER_URL = ActiveMQConnection.DEFAULT_BROKER_URL;// tcp://localhost:61616
	// 消息队列名称
	public static final String DESTINATION = "hoo.mq.queue";
	public static final String USERNAME = ActiveMQConnection.DEFAULT_USER;
	public static final String PASSWORD = ActiveMQConnection.DEFAULT_PASSWORD;

	/**
	 * 发送消息
	 * 
	 * @param session
	 * @param sender
	 * @throws Exception
	 */
	public static void sendMessage(QueueSession session, QueueSender sender) throws Exception {
			for (int i = 0; i < SEND_SEND; i++) {

	String message = "消息发送者 : " + i;
	MapMessage map = session.createMapMessage();
	map.setString("text", message);
	map.setLong("time", System.currentTimeMillis());
	System.out.println(map);
	sender.send(map);

	}
	}

	public static void main(String[] args) {
			QueueConnection connection = null;
			QueueSession session = null;

	try {

	// 创建链接工厂
	QueueConnectionFactory factory = new ActiveMQConnectionFactory(USERNAME, PASSWORD, BROKER_URL);
	// 通过工厂创建一个链接.
	connection = factory.createQueueConnection();
	// 启动链接
	connection.start();
	// 创建一个session会话
	session = connection.createQueueSession(Boolean.TRUE, Session.AUTO_ACKNOWLEDGE);
	// 创建一个消息队列
	Queue queue = session.createQueue(DESTINATION);
	// 创建消息发送者
	QueueSender sender = session.createSender(queue);
	// 设置持久模式
	sender.setDeliveryMode(DeliveryMode.NON_PERSISTENT);
	// 发送消息
	sendMessage(session, sender);
	// 提交会话
	session.commit();

	} catch (Exception e) {
			e.printStackTrace();
	} finally {
			try {
					if (null != session) {
							session.close();
					}
					if (null != connection) {
							connection.close();
					}
			} catch (Exception e) {
					e.printStackTrace();
			}
	}
	}
}



package com.activemq.activemq1_queue;

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.QueueReceiver;
import javax.jms.QueueSession;
import javax.jms.Session;

import org.apache.activemq.ActiveMQConnection;

/**
* 消息接收者
* 
* @author CYX
* @time 2016年12月19日下午7:56:45
*/
public class QueueReceiverMessage {

	public static final String USERNAME = ActiveMQConnection.DEFAULT_USER;
	public static final String PASSWORD = ActiveMQConnection.DEFAULT_PASSWORD;
	public static final String BROKER = ActiveMQConnection.DEFAULT_BROKER_URL;
	public static final String TARGET = "hoo.mq.queue";

	public static void main(String[] args) throws Exception {

	QueueConnection connection = null;
	QueueSession session = null;

	try {
			// 创建链接工厂
			QueueConnectionFactory factory = new org.apache.activemq.ActiveMQConnectionFactory(USERNAME, PASSWORD, BROKER);
			// 通过工厂创建一个链接
			connection = factory.createQueueConnection();
			// 启动链接
			connection.start();
			// 创建一个session会话
			session = connection.createQueueSession(Boolean.TRUE, Session.AUTO_ACKNOWLEDGE);
			// 创建一个消息队列
			Queue queue = session.createQueue(TARGET);
			// 创建消息接收者
			QueueReceiver receiver = session.createReceiver(queue);

	receiver.setMessageListener(new MessageListener() {

	@Override
	public void onMessage(Message msg) {
			if (null != msg) {
					MapMessage map = (MapMessage) msg;
					try {
							System.out.println(map.getLong("time") + " 接收 : " + map.getString("text"));
					} catch (Exception e) {
							e.printStackTrace();
					}
			}
	}
	});
	
	Thread.sleep(1000 * 100);
	session.commit();
	
	} catch (Exception e) {
			e.printStackTrace();
	} finally {
			if (null != session) {
					session.close();
			}
			if (null != connection) {
					connection.close();
			}
	}
	}
}




ActiveMQ服务器工作模型

通过ActiveMQ消息交换信息.消息生产者将消息发送至消息服务,消息消费者则从消息服务接收这些消息。这些消息传送操作是使用一组实现 ActiveMQ应用编程接口 (API) 的对象来执行的。

ActiveMQ客户端使用 ConnectionFactory 对象创建一个连接,向消息服务发送消息以及从消息服务接收消息均是通过此连接来进行。

Connection 是客户端与消息服务的活动连接。创建连接时,将分配通信资源以及验证客户端。这是一个相当重要的对象,大多数客户端均使用一个连接来进行所有的消息传送。连接用于创建会话。

Session 是一个用于生成和使用消息的单线程上下文。它用于创建发送的生产者和接收消息的消费者,并为所发送的消息定义发送顺序。会话通过大量确认选项或通过事务来支持可靠传送。

客户端使用 MessageProducer 向指定的物理目标(在 API 中表示为目标身份对象)发送消息。

生产者可指定一个默认传送模式(持久性消息与非持久性消息)、优先级和有效期值,以控制生产者向物理目标发送的所有消息。


同样,客户端使用 MessageConsumer 对象从指定的物理目标(在 API 中表示为目标对象)接收消息。

消费者可使用消息选择器,借助它,消息服务可以只向消费者发送与选择标准匹配的那些消息。  消费者可以支持同步或异步消息接收。异步使用可通过向消费者注册 MessageListener 来实现。

当会话线程调用 MessageListener 对象的 onMessage 方法时,客户端将使用消息。




ActiveMQ消息传送模型

ActiveMQ 支持两种截然不同的消息传送模型:PTP(即点对点模型)和Pub/Sub(即发布 /订阅模型),分别称作:PTP Domain 和Pub/Sub Domain。


1) PTP(使用Queue 即队列目标) 

消息从一个生产者传送至一个消费者。在此传送模型中,目标是一个队列。

消息首先被传送至队列目标,然后根据队列传送策略,从该队列将消息传送至向此队列进行注册的某一个消费者,一次只传送一条消息。

可以向队列目标发送消息的生产者的数量没有限制,但每条消息只能发送至、并由一个消费者成功使用。

如果没有已经向队列目标注册的消费者,队列将保留它收到的消息,并在某个消费者向该队列进行注册时将消息传送给该消费者。


2) Pub/Sub(使用 Topic即主题目标) 

消息从一个生产者传送至任意数量的消费者。在此传送模型中,目标是一个主题。

消息首先被传送至主题目标,然后传送至所有已订阅此主题的活动消费者。

可以向主题目标发送消息的生产者的数量没有限制,并且每个消息可以发送至任意数量的订阅消费者。

主题目标也支持持久订阅的概念。持久订阅表示消费者已向主题目标进行注册,但在消息传送时此消费者可以处于非活动状态。

当此消费者再次处于活动状态时,它将接收此信息。如果没有已经向主题目标注册的消费者,主题不保留其接收到的消息,除非有非活动消费者注册了持久订阅。




ActiveMQ消息选择器( ?? ) 

ActiveMQ提供了一种机制,使用它,消息服务可根据消息选择器中的标准来执行消息过滤。

生产者可在消息中放入应用程序特有的属性,而消费者可使用基于这些属性的选择标准来表明对消息是否感兴趣。

这就简化了客户端的工作,并避免了向不需要这些消息的消费者传送消息的开销。

然而,它也使得处理选择标准的消息服务增加了一些额外开销。 

消息选择器是用于MessageConsumer的过滤器,可以用来过滤传入消息的属性和消息头部分(但不过滤消息体),并确定是否将实际消费该消息。

消息选择器是一些字符串,它们基于某种语法,而这种语法是SQL-92的子集。可以将消息选择器作为MessageConsumer 创建的一部分。



ActiveMQ消息签收

在不带事务的 Session 中,一条消息何时和如何被签收取决于Session的设置。


1. Session.AUTO_ACKNOWLEDGE 

1) 当客户端从 receive 或 onMessage成功返回时,Session 自动签收客户端的这条消息的收条。在AUTO_ACKNOWLEDGE的 Session 中,同步接收 receive是上述三个阶段的一个例外,在这种情况下,收条和签收紧随在处理消息之后发生。


2. Session.CLIENT_ACKNOWLEDGE 

1) 客户端通过调用消息的 acknowledge 方法签收消息。在这种情况下,签收发生在 Session 层面:签收一个已消费的消息会自动地签收这个 Session 所有已消费消息的收条。


3. Session.DUPS_OK_ACKNOWLEDGE 

1) 此选项指示 Session 不必确保对传送消息的签收。它可能引起消息的重复,但是降低了 Session 的开销,所以只有客户端能容忍重复的消息,才可使用(如果ActiveMQ 再次传送同一消息,那么消息头中的JMSRedelivered 将被设置为 true)。


客户端成功接收一条消息的标志是这条消息被签收。成功接收一条消息一般包括如下三个阶段:

a) 客户端接收消息

b) 客户端处理消息

c) 消息被签收

签收可以由 ActiveMQ发起,也可以由客户端发起,取决于 Session 签收模式的设置。 在带事务的 Session 中,签收自动发生在事务提交时。如果事务回滚,所有已经接收的消息将会被再次传送。 




ActiveMQ优先级

通常,可以确保将单个会话向目标发送的所有消息按其发送顺序传送至消费者。然而,如果为这些消息分配了不同的优先级,消息传送系统将首先尝试传送优先级较高的消息

有两种方法设置消息的优先级 :

1. 使用setDeliveryMode方法,这样所有的消息都采用此传送模式.

2. 使用send方法为每一条消息设置传送模式.


消息优先级从 0-9 十个级别,0-4 是普通消息,5-9 是加急消息。如果不指定优先级,则默认为 4。

JMS 不要求严格按照这十个优先级发送消息,但必须保证加急消息要先于普通消息到达。




ActiveMQ消息过期设置

允许消息过期 。默认情况下,消息永不会过期。如果消息在特定周期内失去意义,那么可以设置过期时间。


有两种方法设置消息的过期时间,时间单位为毫秒:

1. 使用setTimeTolive方法为所有的消息设置过期时间

2. 使用send方法为每一条消息设置过期时间.

消息过期时间,send 方法中的 timeToLive 值加上发送时刻的 GMT 时间值。如果timeToLive 值等于零,则 JMSExpiration 被设为零, 表示该消息永不过期。

如果发送后,在消息过期时间之后消息还没有被发送到目的地,则该消息被清除。 




ActiveMQ持久订阅设置

通过为发布者设置 PERSISTENT传送模式,为订阅者时使用持久订阅,这样可以保证 Pub/Sub 程序接收所有发布的消息。 


消息订阅分为非持久订阅(non-durable subscription)和持久订阅(durable subscription),

非持久订阅只有当客户端处于激活状态,也就是和 ActiveMQ 保持连接状态才能收到发送到某个主题的消息,而当客户端处于离线状态,这个时间段发到主题的消息将会丢失,永远不会收到。

持久订阅时,客户端向ActiveMQ 注册一个识别自己身份的 ID,当这个客户端处于离线时,ActiveMQ会为这个 ID 保存所有发送到主题的消息,当客户端再次连接到ActiveMQ 时, 会根据自己的 ID 得到所有当自己处于离线时发送到主题的消息。持久订阅会增加开销,同一时间在持久订阅中只有一个激活的用户。


建立持久订阅的步骤 : 

1. 为连接设置一个客户ID.

2. 为订阅的主题指定一个订阅名称.

上述组合必须唯一.




ActiveMQ异步发送消息

ActiveMQ支持生产者以同步或异步模式发送消息。使用不同的模式对 send方法的 反应时间有巨大的影响,反映时间是衡量ActiveMQ 吞吐量的重要因素,使用异步发送.

可以提高系统的性能。 在默认大多数情况下,AcitveMQ是以异步模式发送消息。例外的情况:在没有使用事务的情况下,生产者以 PERSISTENT传送模式发送消息。

在这种情况下,send方法都是同步的,并且一直阻塞直到 ActiveMQ发回确认消息:消息已经存储在持久性数据存储中。

这种确认机制保证消息不会丢失,但会造成生产者阻塞从而影响反应时间。


高性能的程序一般都能容忍在故障情况下丢失少量数据。如果编写这样的程序,可以通过使用异步发送来提高吞吐量(甚至在使用PERSISTENT 传送模式的情况下)。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值