当前页面: 开发资料首页 → J2EE 专题 → 消息中间件和消息服务(JMS)
摘要: 消息中间件和消息服务(JMS)
当前,CORBA、DCOM、RMI等RPC中间件技术已广泛应用于各个领域。但是面对规模和复杂度都越来越高的分布式系统,这些技术也显示出其局限性:(1)同步通信:客户发出调用后,必须等待服务对象完成处理并返回结果后才能继续执行;(2)客户和服务对象的生命周期紧密耦合:客户进程和服务对象进程都必须正常运行;如果由于服务对象崩溃或者网络故障导致客户的请求不可达,客户会接收到异常;(3)点对点通信:客户的一次调用只发送给某个单独的目标对象。
面向消息的中间件(Message Oriented Middleware,MOM)较好的解决了以上问题。发送者将消息发送给消息服务器,消息服务器将消息存放在若干队列中,在合适的时候再将消息转发给接收者。这种模式下,发送和接收是异步的,发送者无需等待;二者的生命周期未必相同:发送消息的时候接收者不一定运行,接收消息的时候发送者也不一定运行;一对多通信:对于一个消息可以有多个接收者。
已有的MOM系统包括IBM的MQSeries、Microsoft的MSMQ和BEA的MessageQ等。由于没有一个通用的标准,这些系统很难实现互操作和无缝连接。Java Message Service(JMS)是SUN提出的旨在统一各种MOM系统接口的规范,它包含点对点(Point to Point,PTP)和发布/订阅(Publish/Subscribe,pub/sub)两种消息模型,提供可靠消息传输、事务和消息过滤等机制。
1.JMS
JAVA 消息服务(JMS)定义了Java 中访问消息中间件的接口。JMS 只是接口,并没有给予实现,实现JMS 接口的消息中间件称为JMS Provider,iLink实现了JMS接口,用户可以通过使用JMS接口,在iLink中进行JMS编程。 iLink支持JMS1.0.2版本。
2.JMS接口描述
JMS 支持两种消息类型PTP 和Pub/Sub,分别称作:PTP Domain 和Pub/Sub Domain,这两种接口都继承统一的JMS父接口,JMS 主要接口如下所示:
<table cellSpacing=0 cellPadding=0 width="97%" align=left border=0 NOF="TE">
MS父接口
</td><td width="29%">PTP
</td><td width="48%">Pub/Sub
</td></tr><tr><td width="22%">ConnectionFactory
</td><td width="29%">QueueConnectionFactory
</td><td width="48%">TopicConnectionFactory
</td></tr><tr><td width="22%">Connection
</td><td width="29%">QueueConnection
</td><td width="48%">TopicConnection
</td></tr><tr><td width="22%">Destination
</td><td width="29%">Queue
</td><td width="48%">Topic
</td></tr><tr><td width="22%">Session
</td><td width="29%">QueueSession
</td><td width="48%">TopicSession
</td></tr><tr><td width="22%">MessageProducer
</td><td width="29%">QueueSender
</td><td width="48%">TopicPublisher
</td></tr><tr><td width="22%">MessageConsumer
</td><td width="29%">QueueReceiver,QueueBrowser
</td><td width="48%">TopicSubscriber
</td></tr></table></td></tr></table>
ConnectionFactory :连接工厂,JMS 用它创建连接
Connection :JMS 客户端到JMS Provider 的连接
Destination :消息的目的地
Session: 一个发送或接收消息的线程
MessageProducer: 由Session 对象创建的用来发送消息的对象
MessageConsumer: 由Session 对象创建的用来接收消息的对象
3.JMS消息模型
JMS 消息由以下几部分组成:消息头,属性,消息体。
3.1 消息头(Header) - 消息头包含消息的识别信息和路由信息,消息头包含一些标准的属性如:JMSDestination,JMSMessageID 等。
<table cellSpacing=0 cellPadding=0 width="97%" align=left border=0 NOF="TE">
<tr><td><table id=Table4 cellSpacing=3 cellPadding=1 width="100%" border=1><tr><td width="25%" height=16>消息头
</td><td width="74%">由谁设置
</td></tr><tr><td width="25%">JMSDestination
</td><td width="74%">send 或 publish 方法
</td></tr><tr><td width="25%">JMSDeliveryMode
</td><td width="74%">send 或 publish 方法
</td></tr><tr><td width="25%">JMSExpiration
</td><td width="74%">send 或 publish 方法
</td></tr><tr><td width="25%">JMSPriority
</td><td width="74%">send 或 publish 方法
</td></tr><tr><td width="25%">JMSMessageID
</td><td width="74%">send 或 publish 方法
</td></tr><tr><td width="25%">JMSTimestamp
</td><td width="74%">send 或 publish 方法
</td></tr><tr><td width="25%">JMSCorrelationID
</td><td width="74%">客户
</td></tr><tr><td width="25%">JMSReplyTo
</td><td width="74%">客户
</td></tr><tr><td width="25%">JMSType
</td><td width="74%">客户
</td></tr><tr><td width="25%">JMSRedelivered
</td><td width="74%">JMS Provider
</td></tr></table></td></tr></table>
3.2 属性(Properties) - 除了消息头中定义好的标准属性外,JMS 提供一种机制增加新属性到消息头中,这种新属性包含以下几种:
1. 应用需要用到的属性;
2. 消息头中原有的一些可选属性;
3. JMS Provider 需要用到的属性。
标准的JMS 消息头包含以下属性:
<table cellSpacing=0 cellPadding=0 width="97%" align=left border=0 NOF="TE">
JMSDestination
</td><td width="74%">消息发送的目的地
</td></tr><tr><td width="25%">JMSDeliveryMode
</td><td width="74%">传递模式, 有两种模式: PERSISTENT 和NON_PERSISTENT,PERSISTENT 表示该消息一定要被送到目的地,否则会导致应用错误。NON_PERSISTENT 表示偶然丢失该消息是被允许的,这两种模式使开发者可以在消息传递的可靠性和吞吐量之间找到平衡点。
</td></tr><tr><td width="25%">JMSMessageID
</td><td width="74%">唯一识别每个消息的标识,由JMS Provider 产生。
</td></tr><tr><td width="25%">JMSTimestamp
</td><td width="74%">一个消息被提交给JMS Provider 到消息被发出的时间。
</td></tr><tr><td width="25%">JMSCorrelationID
</td><td width="74%">用来连接到另外一个消息,典型的应用是在回复消息中连接到原消息。
</td></tr><tr><td width="25%">JMSReplyTo
</td><td width="74%">提供本消息回复消息的目的地址
</td></tr><tr><td width="25%">JMSRedelivered
</td><td width="74%">如果一个客户端收到一个设置了JMSRedelivered 属性的消息,则表示可能该客户端曾经在早些时候收到过该消息,但并没有签收(acknowledged)。
</td></tr><tr><td width="25%">JMSType
</td><td width="74%">消息类型的识别符。
</td></tr><tr><td width="25%">JMSExpiration
</td><td width="74%">消息过期时间,等于QueueSender 的send 方法中的timeToLive 值或TopicPublisher 的publish 方法中的timeToLive 值加上发送时刻的GMT 时间值。如果timeToLive值等于零,则JMSExpiration 被设为零,表示该消息永不过期。如果发送后,在消息过期时间之后消息还没有被发送到目的地,则该消息被清除。
</td></tr><tr><td width="25%">JMSPriority
</td><td width="74%">消息优先级,从0-9 十个级别,0-4 是普通消息,5-9 是加急消息。JMS 不要求JMS Provider 严格按照这十个优先级发送消息,但必须保证加急消息要先于普通消息到达。
</td></tr></table></td></tr></table>
3.3 消息体(Body) - JMS API 定义了5种消息体格式,也叫消息类型,你可以使用不同形式发送接收数据并可以兼容现有的消息格式,下面描述这5种类型:
<table cellSpacing=0 cellPadding=0 width="96%" align=left border=0 NOF="TE">
消息类型
</td><td width="73%">消息体
</td></tr><tr><td width="26%">TextMessage
</td><td width="73%">java.lang.String对象,如xml文件内容
</td></tr><tr><td width="26%">MapMessage
</td><td width="73%">名/值对的集合,名是String对象,值类型可以是Java任何基本类型
</td></tr><tr><td width="26%">BytesMessage
</td><td width="73%">字节流
</td></tr><tr><td width="26%">StreamMessage
</td><td width="73%">Java中的输入输出流
</td></tr><tr><td width="26%">ObjectMessage
</td><td width="73%">Java中的可序列化对象
</td></tr><tr><td width="26%">Message
</td><td width="73%">没有消息体,只有消息头和属性
</td></tr></table></td></tr></table>名称
</td><td width="73%">描述
</td></tr><tr><td width="26%">Queue
</td><td width="73%">由JMS Provider 管理,队列由队列名识别,客户端可以通过JNDI 接口用队列名得到一个队列对象。
</td></tr><tr><td width="26%">TemporaryQueue
</td><td width="73%">由QueueConnection 创建,而且只能由创建它的QueueConnection 使用。
</td></tr><tr><td width="26%">QueueConnectionFactory
</td><td width="73%">客户端用QueueConnectionFactory 创建QueueConnection 对象。
</td></tr><tr><td width="26%">QueueConnection
</td><td width="73%">一个到JMS PTP provider 的连接,客户端可以用QueueConnection 创建QueueSession 来发送和接收消息。
</td></tr><tr><td width="26%">QueueSession
</td><td width="73%">提供一些方法创建QueueReceiver 、QueueSender、QueueBrowser 和TemporaryQueue。如果在QueueSession 关闭时,有一些消息已经被收到,但还没有被签收(acknowledged),那么,当接收者下次连接到相同的队列时,这些消息还会被再次接收。
</td></tr><tr><td width="26%">QueueReceiver
</td><td width="73%">客户端用QueueReceiver 接收队列中的消息,如果用户在QueueReceiver 中设定了消息选择条件,那么不符合条件的消息会留在队列中,不会被接收到。
</td></tr><tr><td width="26%">QueueSender
</td><td width="73%">客户端用QueueSender 发送消息到队列。
</td></tr><tr><td width="26%">QueueBrowser
</td><td width="73%">客户端可以QueueBrowser 浏览队列中的消息,但不会收走消息。
</td></tr><tr><td width="26%">QueueRequestor
</td><td width="73%">JMS 提供QueueRequestor 类简化消息的收发过程。QueueRequestor 的构造函数有两个参数:QueueSession 和queue,QueueRequestor 通过创建一个临时队列来完成最终的收发消息请求。
</td></tr><tr><td width="26%">可靠性(Reliability)
</td><td width="73%">队列可以长久地保存消息直到接收者收到消息。接收者不需要因为担心消息会丢失而时刻和队列保持激活的连接状态,充分体现了异步传输模式的优势。
</td></tr></table></td></tr></table>6. PUB/SUB模型
JMS Pub/Sub 模型定义了如何向一个内容节点发布和订阅消息,这些节点被称作主题(topic)。
主题可以被认为是消息的传输中介,发布者(publisher)发布消息到主题,订阅者(subscribe) 从主题订阅消息。主题使得消息订阅者和消息发布者保持互相独立,不需要接触即可保证消息的传送。
下面描述JMS Pub/Sub 模型中的主要概念和对象:
<table cellSpacing=0 cellPadding=0 width="98%" align=left border=0 NOF="TE">
名称
</td><td width="72%">描述
</td></tr><tr><td width="27%">订阅(subscription)
</td><td width="72%">消息订阅分为非持久订阅(non-durable subscription)和持久订阅(durable subscrip-tion),非持久订阅只有当客户端处于激活状态,也就是和JMS Provider 保持连接状态才能收到发送到某个主题的消息,而当客户端处于离线状态,这个时间段发到主题的消息将会丢失,永远不会收到。持久订阅时,客户端向JMS 注册一个识别自己身份的ID,当这个客户端处于离线时,JMS Provider 会为这个ID 保存所有发送到主题的消息,当客户再次连接到JMS Provider时,会根据自己的ID 得到所有当自己处于离线时发送到主题的消息。
</td></tr><tr><td width="27%">Topic
</td><td width="72%">主题由JMS Provider 管理,主题由主题名识别,客户端可以通过JNDI 接口用主题名得到一个主题对象。JMS 没有给出主题的组织和层次结构的定义,由JMS Provider 自己定义。
</td></tr><tr><td width="27%">TemporaryTopic
</td><td width="72%">临时主题由TopicConnection 创建,而且只能由创建它的TopicConnection 使用。临时主题不能提供持久订阅功能。
</td></tr><tr><td width="27%">TopicConnectionFactory
</td><td width="72%">客户端用TopicConnectionFactory 创建TopicConnection 对象。
</td></tr><tr><td width="27%">TopicConnection
</td><td width="72%">TopicConnection 是一个到JMS Pub/Sub provider 的连接,客户端可以用TopicConnection创建TopicSession 来发布和订阅消息。
</td></tr><tr><td width="27%">TopicSession
</td><td width="72%">TopicSession 提供一些方法创建TopicPublisher、TopicSubscriber、TemporaryTopic 。它还提供unsubscribe 方法取消消息的持久订阅。
</td></tr><tr><td width="27%">TopicPublisher
</td><td width="72%">客户端用TopicPublisher 发布消息到主题。
</td></tr><tr><td width="27%">TopicSubscriber
</td><td width="72%">客户端用TopicSubscriber 接收发布到主题上的消息。可以在TopicSubscriber 中设置消息过滤功能,这样,不符合要求的消息不会被接收。
</td></tr><tr><td width="27%">Durable TopicSubscriber
</td><td width="72%">如果一个客户端需要持久订阅消息,可以使用Durable TopicSubscriber,TopSession 提供一个方法createDurableSubscriber创建Durable TopicSubscriber 对象。
</td></tr><tr><td width="27%">恢复和重新派送(Recovery and Redelivery)
</td><td width="72%">非持久订阅状态下,不能恢复或重新派送一个未签收的消息。只有持久订阅才能恢复或重新派送一个未签收的消息。
</td></tr><tr><td width="27%">TopicRequestor
</td><td width="72%">JMS 提供TopicRequestor 类简化消息的收发过程。TopicRequestor 的构造函数有两个参数:TopicSession 和topic。TopicRequestor 通过创建一个临时主题来完成最终的发布和接收消息请求。
</td></tr><tr><td width="27%">可靠性(Reliability)
</td><td width="72%">当所有的消息必须被接收,则用持久订阅模式。当丢失消息能够被容忍,则用非持久订阅模式。
</td></tr></table></td></tr></table>
7. 开发JMS的步骤
广义上说,一个JMS 应用是几个JMS 客户端交换消息,开发JMS 客户端应用由以下几步构成:
用JNDI 得到ConnectionFactory 对象;
用JNDI 得到目标队列或主题对象,即Destination 对象;
用ConnectionFactory 创建Connection 对象;
用Connection 对象创建一个或多个JMS Session;
用Session 和Destination 创建MessageProducer 和MessageConsumer;
通知Connection 开始传递消息。
更详细的信息请参考 Java Message Specification.