- 浏览: 832825 次
- 性别:
- 来自: 厦门
文章分类
- 全部博客 (363)
- 2010年3月 (3)
- 2010年4月 (3)
- Java (116)
- ExtJs (22)
- EJB3.0 (11)
- JQuery (28)
- SqlServer (5)
- Oracle (17)
- hibernate (21)
- struts2 (14)
- php (10)
- JavaScript (11)
- jbpm (6)
- spring (24)
- lucene (2)
- ibatis (7)
- C# (8)
- mysql (11)
- json (3)
- webservice (3)
- 设计模式 (1)
- jdbc (1)
- servlet (2)
- ice (6)
- 日常软件问题 (2)
- 生活 (2)
- iphone (1)
- rest (3)
- ruby (2)
- linux (13)
- quartz (1)
- poi (1)
- redis (13)
- memcached (4)
- nosql (2)
- tomcat调优 (1)
- 项目管理 (0)
最新评论
-
天使建站:
jquery里和数组相关的操作 附带具体的实例 ...
jquery对象数组 -
Cy0941:
$('#formId').form('submit',...) ...
easyui的form表单提交处理 -
shmily2038:
swifth 写道楼主,,你的命令写错啦,,[root@ser ...
centos直接yum安装nginx -
swifth:
楼主,,你的命令写错啦,,[root@server ~]# y ...
centos直接yum安装nginx -
随遇而安DXX:
...
REST
JMS简介:一种应用于异步消息传递的标准API,JMS也是应用于程序间通讯的。但是,JMS与其他机制有所不同,主要表现在系统间传递信息的方式,见PPT1-2。简介传送也是JMS的关键。当一个应用程序通过JMS向另一个应用程序发送消息时,两个程序之间并没有直接的连接。发送应用程序会将消息交给一个服务,由服务确保将消息投递给接收应用程序。在JMS中有两个主要的概念:消息中介(message broker)和消息目标(destination)。当应用程序发送消息时,会将消息交给一个消息中介。消息中介实际上就是JMS版本的邮局。消息中介可以确保消息被投递到指定的消息目标,同时可以释放发送者,使其能够进行其他的业务。 在JMS中,每条消息带有一个消息目标。消息目标就好像一个邮箱,可以将消息放入这个邮箱,直到有人将它们取出。消息目标只关心消息应该从哪里获得--而不是由谁来获得。在JMS中,有两种消息目标类型:队列和主题。分别应用于队列的点对点模型或应用于主题的发布--订阅模型。
点对点消息传递模型:在点对点模型中,每个消息都有一个发送者和一个接收者,见PPT3.在JMS中如果有多个接收者监听队列,就没办法直到某条特定的消息会被哪个接收者处理。这种不确定性实际上有很多好处,因为它可以让程序之用为队列添加监听器就能增大消息处理的能力。
发布--订阅消息传递模型:在此模型中,消息会被发送给一个主题。像使用队列一样,可以让多个接收者监听一个主题。但是,与队列不同的是消息不再被只投递给一个接收者,所有主题的订阅者都会收到消息,见PPT4。
JMS的优点:不用等待--当使用JMS发送消息时,客户端不必等待消息被处理,甚至是被投递。客户端只需要将消息发给消息中介,就可以确信消息会被投递到适当的目标。因为不必等待,客户端就可以执行其他的任务。由于这种方法可以大大地节省时间,客户端的性能能够极大地提高。面向消息--与RFC通信面向方法调用不同,使用JMS发送消息是以数据位中心的。这意味着客户端不用固定搭配特定的方法符号。任何队列或主题订阅者都可以处理由客户端发送来的消息。客户端不必了解服务的任何规范。位置独立--JMS客户端不必知道由谁来处理他们的消息,或者服务的位置在哪里。客户端值需要了解需要通过哪个队列或主题发送消息。因此,只要能够从队列或主题获取消息,JMS客户端就不用在意服务的位置在哪里。确保投送--当使用JMS发送消息时,客户端能够确保消息被投递。即使在消息发送时服务无法使用,消息也会被储存起来,直到服务重新可以使用为止。
在Spring中安装ActiveMQ:它是一个开源消息中介(Apache的一个子项目),也是应用JMS异步消息传递的很好选择。需要从www.activemq.org下载包。在bin目录中,可以找到一个用于启动ActiveMQ的脚本:Unix用户的activemq或者Windows用户的activemq.bat。运行这个脚本,等ActiveMq启动后,就可以使用它进行中介服务了。在所有的示例中,都需要通过JMS连接工厂通过消息中介发送消息。我们选择了ActiveMQ作为消息中介,因此必须配置JMS连接工厂,使它能够了解如何连接到ActiveMQ,ActiveMQConnectionFactory是连接ActiveMQ的JMS连接工厂,在Spring中配置如下:
<bean id ="connectionFactory" class="org.apache.activemq.ActiveMQConnectionFactory" >
<property name="brokerURL" value="tcp://localhost:61616">
</bean>
声明一个ActiveMQ队列:
<bean id="rantzDestination" class="org.apache.activemq.command.ActiveMQQueue">
<constructor-arg index="0" value="rantz.marketing.queue" />
</bean>
声明一个ActiveMQ主题:
<bean id="rantzDestination" class="org.apache.activemq.command.ActiveMQTopic">
<constructor-arg index="0" value="rantz.marketing.topic" />
</bean>
协同使用JMS和Spring:JMS为Java爱好者提供了一个与消息中介进行交互,以及发送和接收消息的标准API,而且每一个消息中介的实现都会支持JMS。因此你不必由于消息中介的不同而学习不同的消息传递的API。虽然JMS为所有的消息中介提供了统一接口,但这种接口用起来不是十分便利(见PPT5-6)
使用JMS模板:JmsTemplate是Spring消除冗长和重复JMS代码的解决方案。JmsTemplate可以创建连接,获取会话,以及发送和接收消息。它使你可以专注于构建要发送的消息或者处理收到的消息。另外,JmsTemplate可以处理任何被抛出的JMSException。如果JMSException在JmsTemplate工作中被抛出,JmsTemplate将捕获这个异常,并且用一个未受查的JmsException的子类再次抛出它。见PPT7
置入JmsTemplate模板:
<bean id ="jmsTemplate" class="org.springframework.jms.core.JmsTemplate">
<property name="connectionFactory" ref="connectionFactory">
</bean>
发送消息:在RoadRantz端,我们使用JmsTemplate向RoandRantz市场发送驾驶员信息。RantzMarketingGatewayImpl是RoadRantz与市场系统进行交互的类。见PPT8,因此,当我们在Spring中配置RantzMarKetingGatewayImpl类时,必须置入jmsTemplate和rantzDestination Bean的参考:
<bean id="marketingGateway" class="com.roadrantz.marketing.RantzMarketingGatewayImpl">
<property name="jmsTemplate" ref ="jmsTemplate" />
<property name="destination" ref="rantzDestination" />
</bean>
设置默认目标:如果每次发送消息都指定一个目标,不如为JmsTemplate置入一个默认目标:
<bean id ="jmsTemplate" class="org.springframework.jms.core.JmsTemplate">
<property name="connectionFactory" ref="connectionFactory" />
<property name="defaultDestination" ref="rantzDestination" />
</bean>
现在,调用JmsTemplate的send()方法就可以去掉第一个参数了。这种send()方法的形式只带有一个MessageCreator。因为没有指定目标,JmsTemplate会假设你要将消息发送给默认目标。所以不必再为RantzMarketingGatewayImpl注入目标了。它的声明很简单:
<bean id="marketingGateway" class="com.roadrantz.marketing.RantzMarketingGatewayImpl">
<property name="jmsTemplate" ref ="jmsTemplate" />
</bean>
消费消息:使用JmsTemplate接收消息十分简单。只需要调用JmsTemplate的receive()方法,如PPT9. 默认情况下,对receive()方法的调用会造成阻塞,直到消息到达目标--如果必要,会永远等待下去。为了避免对消息的内部等待,可以通过配置JmsTemplate时,通过设置receiveTimeout属性来指定接收时间。下面配置的接收超时时间为一分钟(60000毫秒)。
<bean id ="jmsTemplate" class="org.springframework.jms.core.JmsTemplate">
<property name="connectionFactory" ref="connectionFactory" />
<property name="defaultDestination" ref="rantzDestination" />
<property name="receiveTimeout" value="60000" />
</bean>
在PPT9中,receive()方法会从默认目标接收消息。如果你希望指定一个目标,可以为其传递一个目标:
MapMessage message = (MapMessage) jmsTemplate.receive(destination);
此外,还可以通过名称指定目标,并且让Spring的目标解析器自动解析这个目标:
MapMessage message = (MapMessage) jmsTemplate.receive("rantz.marketing.queue");
转换消息:为了简化示例,没有在发送和接收消息的代码中添加处理消息转换的代码。但是,如果在程序中的多个位置都需要发送或接收相同的消息,可能就需要通过消息转换器来避免不必要的映射代码。尽管将消息转换代码抽象成自己的工具类并不是很困难,但是你仍然需要明确的调用工具类来进行转换。幸运的是,Spring通过MessageConverter接口提供了对消息转换的支持:
public interface MessageConverter {
public Message toMessage(Object object, Session session);
public Object fromMessage (Message message) ;
}
见PPT10,展示了一个MessageConverter接口的实现MotoristMessageConverter,可以将Motorist对象转换为消息,也可以将消息转换为SpammedMotorist对象。
发送和接收被转换的消息:我们可以在发送消息前不用明确的调用toMessage()方法,只需要调用JmsTemplate的convertAndSend()方法。因此,PPT8中的sendMotoristInfo()方法将变得更加简单:
public void sendMotoristInfo (final Motorist motorist){
jmsTemplate.converAndSend(motorist); } 消息被发向JmsTemplate的默认目标(假设已经指定了一个默认目标)。不过我们也可以在调用convertAndSend()方法时指定一个特定的目标: jmsTemplate.connverAndSend(destination,motorist); 另外,我们也可以通过名称指定目标: jmsTemplate.convertAndSend("rantz.marketing.queue",motorist); 接收端: public SpammedMotorist receiveSpammedMotorist() { return (SpammedMotorist) jmsTemplate.receiveAndConvert(); } 除非有特殊的指定,recieveAndConvert()会从默认目标接收消息。不过,我们也可以通过为receiveAndConvert()方法传递一个参数来选择一个目标: return (SpammedMotorist) jmsTemplate.receiveAndConvert(destination); 或者使用目标的名称: return (SpammedMotorist) jmsTemplate.receiveAndConvert("rantz.marketing.queue");
置入消息转化器: <bean id="motoristConverter" class="com.roadrantz.marketing.MotoristMessageConverter" />
最后,JmsTemplate需要了解这个消息转换器。为了提供消息转换器,我们会将motoristConverter Bean置入到JmsTemplate的messageConverter属性中:
<bean id ="jmsTemplate" class="org.springframework.jms.core.JmsTemplate">
<property name="connectionFactory" ref="connectionFactory" />
<property name="defaultDestination" ref="rantzDestination" />
<property name="messageConverter" ref="motoristConverter" />
</bean>
将Spring的网关支持类应用于JMS: Spring通过提供JdbcDaoSupport使JdbcTemplate的应用比原先变得更简单一些。与此类似,Spring也提供了一个用于JMS网关类的基类JmsGatewaySupport(见PPT11)。 那么JmsGatewaySupport从哪里得到JmsTemplate呢?你可以直接在jmsTemplate属性中注入JmsTemplate,就像处理常规的RantzMarketingGatewayImpl一样。或者,可以为connectionFactory属性置入连接工厂,来满足对JmsTemplate Bean的所有需求:
<bean id = "marketingGateway" class="com.roadrantz.marketing.RantzMarketingGatewayImple">
<property name="connectionFactory " ref="connectionFactory" />
</bean>
当这种方法配置时,JmsGateSupport将基于被注入的连接工厂自动创建一个JmsTemplate对象。因此不必在Spring中声明JmsTemplate Bean。 在直接将连接工厂置入到网关之前,你应该了解这种方法有两个缺点:
1、只能在JmsTemplate上指定默认目标。如果JmsGateSupport创建了自己的JmsTemplate,就没有机会再指定默认的目标了。你必须在调用send()或receive()时明确地选择一个目标。
2、只能将消息转换器置入到JmsTemplate中。如果JmsGatewaySupport创建了自己JmsTemplate,将不能使用消息转换器。因此,必须明确地在网关代码中处理消息的转换。
创建消息驱动POJO:EJB2.0规范的其中一个重要内容是包含了消息驱动Bean(MDB)。MDB是可以异步处理消息的EJB。换句话说,MDB会将JMS目标中的消息作为事件来响应。这与同步消息接收者在消息可用前进行阻塞正好相反。MDB是EJB的一个两点。在EJB3的规范中,MDB被简化了,使其更像POJO。不再需要实现MessageDrivenBean接口,只需要实现常规的java.jms.MessageListener接口并用@MessageDriven注释MDB。Spring2.0通过自己的消息驱动Bean形式来满足消息的异步消费需求。
创建消息监听器:MarketingMdp不必实现MessageDrivenBean接口的世界是多么简单,见PPT12。MarketingMdp本身并不做些什么。它有一个实际处理消息的onMessage()方法。不过要先在Spring中配置一下:
<bean id="rantzMdp" class="com.roadrantz.marketing.MarketingMdp" />
EJB3.0MDB会使用@MessageDriven注释通知容器这是一个MDB。但是,在Spring中,我们会通过将其注入到一个消息监听容器来指示这个Bean是一个MDP。
包含消息监听器:消息监听器容器是一个用于查看JMS目标,等待消息到达的特殊Bean。一旦消息到达,它就可以获取到消息,并通过onMessage()方法将消息传递给一个MessageListener的实现。因为MarketingMdp类实现了MessageListener接口,所以消息监听器容器就准备完毕了。见PPT13. 顾名思义,SimpleMessageListenerContainer是最简单的消息监听器容器,可以按下面的方法在Spring中进行配置:
<bean class="org.springframework.jms.listener.SimpleMessageListenerContainer">
<property name="connectionFactory" ref="connectionFactory" />
<property name="destination" ref="rantzDestination" />
<property name="messageListener" ref="rantzMdp" />
</bean>
对于messageListener属性,我们为其置入了对MDP实现的引用,这样,onMessage()方法将可以被用来接收消息。
使用事务性的MDP:如果收到的一个消息在事务中,则应该使用DefaultMessageListenerContainer:
<bean class="org.springframework.jms.listener.DefaultMessageListenerContainer">
<property name="connectionFactory" ref="connectionFactory" />
<property name="destination" ref="rantzDestination" />
<property name="messageListener" ref="rantzMdp" />
<property name="transactionManager" ref="jmsTransactionManager" />
</bean> 如果事务性需求比较简单,JmsTransactionManager将按如下方法配置:
<bean id="jmsTransactionManager" class="org.springframework.jms.connection.JmsTransactionManager">
<property name="connectionFactory" ref="connectionFactory" />
</bean> 需要提醒的是,tansactionManger属性是可选的。如果不注入事务管理器,MDP就不是事务性的。
编写纯POJO MDP:如果消息监听器容器的messageListener属性被注入了MessageListener的实现,它就能够知道在消息到达时应该调用onMessage()方法。幸运的是,Spring提供了一个替代的MessageListenerAdapter。它是一个MessageListener,可以委派Bean和你选择的方法,见PPT14. 如果不将自己的MessageListener的实现注入到消息监听器容器中,你可以置入到MessageListenerAdapter中:
<bean class="org.springframework.jms.listener.SimpleMessageListenerContainer">
<property name="connectionFactory" ref="connectionFactory" />
<property name="destination" ref="rantzDestination" />
<property name="messageListener" ref ="purePojoMdp" />
</bean> 因为配置了purePojoMdp Bean,所以它是一个MessageListenerAdapter:
<bean id="purePojoMdp" class="org.springframework.jms.listener.adapter.MessageListenerAdapter">
<property name="delegate" ref="rantzMdp" />
<property name="defaultListenerMethod" value="processMotoristInfo" />
</bean> 默认情况下,MessageListenerAdapter在消息到达时会调用handleMessage()方法。但是,我们希望MarketingMdp Bean可以通过 processMotoristInfo()方法处理消息,因此将defaultListenerMethod设置为processMotoristInfo. 因为选择了一个特定的被调用方法,所以不需要实现MessageListener或onMessage()方法。因此 MarketingMdp现在将被简化为PPT15。尽管它是一个POJO,对MapMessage的依赖造成了MarketingMdp与JMS的不必要耦合,另外,MapMessage的getString方法还会抛出必须被处理的JMSException。理想情况下,MarketingMdp不应该依赖任何特定框架的类型。 当MessageListenerAdapter接收消息时,它会考虑消息的类型和defaultListenerMethod的值,并且尝试着查找用来调用的监听器方法符号。PPT16描述了MessageListenerAdapter是如何将JMS消息映射到监听器方法参数的。
转换MDP消息:在最新版中,processMotoristInfo()带有的是Map,并且在处理前需要将Map转换为SpammedMotorist。如果在消息到达时,能够直接给processMotoristInfo()方法传递可以处理的SpammedMotorist对象岂不是更好么?Spring消息转换器可以执行消息和特定域Java类型之间的相互转换工作。在PPT10中,已经有一个消息转换器。我们需要做的就是让MessageListenerAdapter能够感知这个消息转换器。MessageListenerAdapter的messageConverter属性可以完成这项工作:
<bean id="purePojoMdp" class="org.springframework.jms.listener.adapter.MessageListenerAdapter">
<property name="delegate" ref="rantzMdp" />
<property name="defaultListenerMethod" value="processMotoristInfo" />
<propert name="messageConverter" ref="motoristConverter" />
</bean> 现在,可以编写最终版的MarketingMdp了。
使用基于消息的RPC:RPC编程模型可以使与远程服务的交互就如同在调用本地对象的方法。是否有某种方法既具有RPC编程模型的简单性,又可以利用异步消息传递的优点呢?
引入Lingo: 它是一种基于Spring的远程调用方法,它在RPC和异步消息传递之间架起了一座桥梁。与使用其他的Spring远程调用方法相同,Lingo提供了一个服务输出器,可以将Bean的功能输出为Lingo服务和客户端代理。 Lingo远程调用会通过JMS队列或主题来承载信息。 尽管Lingo是基于Spring远程调用的,但是它并不是Spring框架的一部分。你可以从Lingo的主页下载Lingo(http://lingo.codehaus.org/Download).
输出服务:在服务端,Lingo提供了一个服务输出器JmsServiceExporter。被Lingo输出的服务是通过JMS来提供服务的,而不是用于直接的RPC访问,见PPT17。下面的XML配置了一个JmsServiceExporter,能够将rantzMdp Bean输出为RPC-over-JMS服务:
<bean id="server" class="org.logicblaze.lingo.jms.JmsServiceExporter">
<property name="connectionFactory" ref="connectionFactory" />
<property name="destination" ref="destination" />
<property name="service" ref="rantzMdp" />
<property name="serviceInterface" value="com.roadrantz.marketing.MarketingService" />
</bean>
service属性被置入了对rantzMdp Bean的引用,即MarketingMdp。最后,应该将定义了服务的接口的类名配置到serviceInterface属性中。我们声明的被输出服务带有MarketingService接口,定义如下:
public interface MarketingService {
void processMotoristInfo (SpammedMotorist motorist );
} 因为我们定义的服务带有MarketingService接口,这就意味着我们应该对MarketingMdp类进行一些小的修改,以便它可以实现MarketingService接口:
public class MarketingMdp implements MarketingService {
public void processMotoristInfo(SpammedMotorist motorist) {
...
}
}
这就是使用Lingo输出服务的全过程。一旦应用程序启动,JmsServiceExporter将会起作用,就可以开始使用它了。现在,让我们再到客户端看看RoadRantz应用程序是如何调用这个被输出的市场服务的。
代理JMS:在RoadRantz应用程序中,每次用户注册并选择愿意接收特定提供商信息时都需要调用processMotoristInfo()方法,因此,我们必须以某种方法将Lingo输出服务的引用置入到RoadRantz应用程序中。
置入JmsProxyFactoryBean:lingo提供了JmsProxyFactoryBean,这是一个代理工厂Bean,可以生成远程Lingo输出服务的代理。如PPT18所示,通过JmsProxyFactoryBean代理的服务是通过JMS目标(队列或主题)访问的,而不是通过TCP/IP。下面声明配置了一个JmsProxyFactoryBean:
<bean id = "marketing" class="org.logicblaze.lingo.jms.JmsProxyFactoryBean">
<property name="connectionFactory" ref="connectionFactory" />
<property name="destination" ref="destination" />
<property name="serviceInterface" value="com.roadrantz.marketing.MarketingService" />
</bean> serviceInterface属性指定了代理需要实现的Java接口。通过这个接口,RoadRantz可以调用processMotoristInfo()方法。 对于配置JmsProxyFactoryBean,最需要注意的事情是不用配置服务的位置信息。这是因为服务的位置并不重要,只需要知道服务在哪里接收"邮件"。事实上,我们还可以使用远程服务的多个实例。如果想建立一个高可靠性的市场服务,可以启动两个或多个实例,让它们都监听相同的目标。每一个实例都能够处理一个请求。其间,客户端并不知道有一个服务池在等待请求进行响应。
进行调用:在JmsProxyFactoryBean被置入后,就可以开始调用远程服务了。我们需要做的是将其置入到RantServiceImpl:
<bean id ="rantService" class="com.roadrantz.service.RantServiceImpl">
<property name="rantDao" ref="rantDao" />
<property name="marketingService" ref="marketing" />
</bean> 接着,我们可以使用它在addMotorist()方法中向市场服务发送SpammedMotorist对象。PPT19展示了为了调用远程市场服务对RantServiceImpl进行的相关修改。 正如你看到的,调用Lingo输出的服务与调用RMI服务、Web服务或在同一个进程中调用其他Bean上的方法是相同的。PPT19中没有任何关于JMS的内容。唯一不同的地方在Spring的配置。通过这种方法,只需要简单的更改Spring的配置就能够在JMS和其他通信机制直接进行切换。
实例:http://blog.csdn.net/pengchua/article/details/4444400
点对点消息传递模型:在点对点模型中,每个消息都有一个发送者和一个接收者,见PPT3.在JMS中如果有多个接收者监听队列,就没办法直到某条特定的消息会被哪个接收者处理。这种不确定性实际上有很多好处,因为它可以让程序之用为队列添加监听器就能增大消息处理的能力。
发布--订阅消息传递模型:在此模型中,消息会被发送给一个主题。像使用队列一样,可以让多个接收者监听一个主题。但是,与队列不同的是消息不再被只投递给一个接收者,所有主题的订阅者都会收到消息,见PPT4。
JMS的优点:不用等待--当使用JMS发送消息时,客户端不必等待消息被处理,甚至是被投递。客户端只需要将消息发给消息中介,就可以确信消息会被投递到适当的目标。因为不必等待,客户端就可以执行其他的任务。由于这种方法可以大大地节省时间,客户端的性能能够极大地提高。面向消息--与RFC通信面向方法调用不同,使用JMS发送消息是以数据位中心的。这意味着客户端不用固定搭配特定的方法符号。任何队列或主题订阅者都可以处理由客户端发送来的消息。客户端不必了解服务的任何规范。位置独立--JMS客户端不必知道由谁来处理他们的消息,或者服务的位置在哪里。客户端值需要了解需要通过哪个队列或主题发送消息。因此,只要能够从队列或主题获取消息,JMS客户端就不用在意服务的位置在哪里。确保投送--当使用JMS发送消息时,客户端能够确保消息被投递。即使在消息发送时服务无法使用,消息也会被储存起来,直到服务重新可以使用为止。
在Spring中安装ActiveMQ:它是一个开源消息中介(Apache的一个子项目),也是应用JMS异步消息传递的很好选择。需要从www.activemq.org下载包。在bin目录中,可以找到一个用于启动ActiveMQ的脚本:Unix用户的activemq或者Windows用户的activemq.bat。运行这个脚本,等ActiveMq启动后,就可以使用它进行中介服务了。在所有的示例中,都需要通过JMS连接工厂通过消息中介发送消息。我们选择了ActiveMQ作为消息中介,因此必须配置JMS连接工厂,使它能够了解如何连接到ActiveMQ,ActiveMQConnectionFactory是连接ActiveMQ的JMS连接工厂,在Spring中配置如下:
<bean id ="connectionFactory" class="org.apache.activemq.ActiveMQConnectionFactory" >
<property name="brokerURL" value="tcp://localhost:61616">
</bean>
声明一个ActiveMQ队列:
<bean id="rantzDestination" class="org.apache.activemq.command.ActiveMQQueue">
<constructor-arg index="0" value="rantz.marketing.queue" />
</bean>
声明一个ActiveMQ主题:
<bean id="rantzDestination" class="org.apache.activemq.command.ActiveMQTopic">
<constructor-arg index="0" value="rantz.marketing.topic" />
</bean>
协同使用JMS和Spring:JMS为Java爱好者提供了一个与消息中介进行交互,以及发送和接收消息的标准API,而且每一个消息中介的实现都会支持JMS。因此你不必由于消息中介的不同而学习不同的消息传递的API。虽然JMS为所有的消息中介提供了统一接口,但这种接口用起来不是十分便利(见PPT5-6)
使用JMS模板:JmsTemplate是Spring消除冗长和重复JMS代码的解决方案。JmsTemplate可以创建连接,获取会话,以及发送和接收消息。它使你可以专注于构建要发送的消息或者处理收到的消息。另外,JmsTemplate可以处理任何被抛出的JMSException。如果JMSException在JmsTemplate工作中被抛出,JmsTemplate将捕获这个异常,并且用一个未受查的JmsException的子类再次抛出它。见PPT7
置入JmsTemplate模板:
<bean id ="jmsTemplate" class="org.springframework.jms.core.JmsTemplate">
<property name="connectionFactory" ref="connectionFactory">
</bean>
发送消息:在RoadRantz端,我们使用JmsTemplate向RoandRantz市场发送驾驶员信息。RantzMarketingGatewayImpl是RoadRantz与市场系统进行交互的类。见PPT8,因此,当我们在Spring中配置RantzMarKetingGatewayImpl类时,必须置入jmsTemplate和rantzDestination Bean的参考:
<bean id="marketingGateway" class="com.roadrantz.marketing.RantzMarketingGatewayImpl">
<property name="jmsTemplate" ref ="jmsTemplate" />
<property name="destination" ref="rantzDestination" />
</bean>
设置默认目标:如果每次发送消息都指定一个目标,不如为JmsTemplate置入一个默认目标:
<bean id ="jmsTemplate" class="org.springframework.jms.core.JmsTemplate">
<property name="connectionFactory" ref="connectionFactory" />
<property name="defaultDestination" ref="rantzDestination" />
</bean>
现在,调用JmsTemplate的send()方法就可以去掉第一个参数了。这种send()方法的形式只带有一个MessageCreator。因为没有指定目标,JmsTemplate会假设你要将消息发送给默认目标。所以不必再为RantzMarketingGatewayImpl注入目标了。它的声明很简单:
<bean id="marketingGateway" class="com.roadrantz.marketing.RantzMarketingGatewayImpl">
<property name="jmsTemplate" ref ="jmsTemplate" />
</bean>
消费消息:使用JmsTemplate接收消息十分简单。只需要调用JmsTemplate的receive()方法,如PPT9. 默认情况下,对receive()方法的调用会造成阻塞,直到消息到达目标--如果必要,会永远等待下去。为了避免对消息的内部等待,可以通过配置JmsTemplate时,通过设置receiveTimeout属性来指定接收时间。下面配置的接收超时时间为一分钟(60000毫秒)。
<bean id ="jmsTemplate" class="org.springframework.jms.core.JmsTemplate">
<property name="connectionFactory" ref="connectionFactory" />
<property name="defaultDestination" ref="rantzDestination" />
<property name="receiveTimeout" value="60000" />
</bean>
在PPT9中,receive()方法会从默认目标接收消息。如果你希望指定一个目标,可以为其传递一个目标:
MapMessage message = (MapMessage) jmsTemplate.receive(destination);
此外,还可以通过名称指定目标,并且让Spring的目标解析器自动解析这个目标:
MapMessage message = (MapMessage) jmsTemplate.receive("rantz.marketing.queue");
转换消息:为了简化示例,没有在发送和接收消息的代码中添加处理消息转换的代码。但是,如果在程序中的多个位置都需要发送或接收相同的消息,可能就需要通过消息转换器来避免不必要的映射代码。尽管将消息转换代码抽象成自己的工具类并不是很困难,但是你仍然需要明确的调用工具类来进行转换。幸运的是,Spring通过MessageConverter接口提供了对消息转换的支持:
public interface MessageConverter {
public Message toMessage(Object object, Session session);
public Object fromMessage (Message message) ;
}
见PPT10,展示了一个MessageConverter接口的实现MotoristMessageConverter,可以将Motorist对象转换为消息,也可以将消息转换为SpammedMotorist对象。
发送和接收被转换的消息:我们可以在发送消息前不用明确的调用toMessage()方法,只需要调用JmsTemplate的convertAndSend()方法。因此,PPT8中的sendMotoristInfo()方法将变得更加简单:
public void sendMotoristInfo (final Motorist motorist){
jmsTemplate.converAndSend(motorist); } 消息被发向JmsTemplate的默认目标(假设已经指定了一个默认目标)。不过我们也可以在调用convertAndSend()方法时指定一个特定的目标: jmsTemplate.connverAndSend(destination,motorist); 另外,我们也可以通过名称指定目标: jmsTemplate.convertAndSend("rantz.marketing.queue",motorist); 接收端: public SpammedMotorist receiveSpammedMotorist() { return (SpammedMotorist) jmsTemplate.receiveAndConvert(); } 除非有特殊的指定,recieveAndConvert()会从默认目标接收消息。不过,我们也可以通过为receiveAndConvert()方法传递一个参数来选择一个目标: return (SpammedMotorist) jmsTemplate.receiveAndConvert(destination); 或者使用目标的名称: return (SpammedMotorist) jmsTemplate.receiveAndConvert("rantz.marketing.queue");
置入消息转化器: <bean id="motoristConverter" class="com.roadrantz.marketing.MotoristMessageConverter" />
最后,JmsTemplate需要了解这个消息转换器。为了提供消息转换器,我们会将motoristConverter Bean置入到JmsTemplate的messageConverter属性中:
<bean id ="jmsTemplate" class="org.springframework.jms.core.JmsTemplate">
<property name="connectionFactory" ref="connectionFactory" />
<property name="defaultDestination" ref="rantzDestination" />
<property name="messageConverter" ref="motoristConverter" />
</bean>
将Spring的网关支持类应用于JMS: Spring通过提供JdbcDaoSupport使JdbcTemplate的应用比原先变得更简单一些。与此类似,Spring也提供了一个用于JMS网关类的基类JmsGatewaySupport(见PPT11)。 那么JmsGatewaySupport从哪里得到JmsTemplate呢?你可以直接在jmsTemplate属性中注入JmsTemplate,就像处理常规的RantzMarketingGatewayImpl一样。或者,可以为connectionFactory属性置入连接工厂,来满足对JmsTemplate Bean的所有需求:
<bean id = "marketingGateway" class="com.roadrantz.marketing.RantzMarketingGatewayImple">
<property name="connectionFactory " ref="connectionFactory" />
</bean>
当这种方法配置时,JmsGateSupport将基于被注入的连接工厂自动创建一个JmsTemplate对象。因此不必在Spring中声明JmsTemplate Bean。 在直接将连接工厂置入到网关之前,你应该了解这种方法有两个缺点:
1、只能在JmsTemplate上指定默认目标。如果JmsGateSupport创建了自己的JmsTemplate,就没有机会再指定默认的目标了。你必须在调用send()或receive()时明确地选择一个目标。
2、只能将消息转换器置入到JmsTemplate中。如果JmsGatewaySupport创建了自己JmsTemplate,将不能使用消息转换器。因此,必须明确地在网关代码中处理消息的转换。
创建消息驱动POJO:EJB2.0规范的其中一个重要内容是包含了消息驱动Bean(MDB)。MDB是可以异步处理消息的EJB。换句话说,MDB会将JMS目标中的消息作为事件来响应。这与同步消息接收者在消息可用前进行阻塞正好相反。MDB是EJB的一个两点。在EJB3的规范中,MDB被简化了,使其更像POJO。不再需要实现MessageDrivenBean接口,只需要实现常规的java.jms.MessageListener接口并用@MessageDriven注释MDB。Spring2.0通过自己的消息驱动Bean形式来满足消息的异步消费需求。
创建消息监听器:MarketingMdp不必实现MessageDrivenBean接口的世界是多么简单,见PPT12。MarketingMdp本身并不做些什么。它有一个实际处理消息的onMessage()方法。不过要先在Spring中配置一下:
<bean id="rantzMdp" class="com.roadrantz.marketing.MarketingMdp" />
EJB3.0MDB会使用@MessageDriven注释通知容器这是一个MDB。但是,在Spring中,我们会通过将其注入到一个消息监听容器来指示这个Bean是一个MDP。
包含消息监听器:消息监听器容器是一个用于查看JMS目标,等待消息到达的特殊Bean。一旦消息到达,它就可以获取到消息,并通过onMessage()方法将消息传递给一个MessageListener的实现。因为MarketingMdp类实现了MessageListener接口,所以消息监听器容器就准备完毕了。见PPT13. 顾名思义,SimpleMessageListenerContainer是最简单的消息监听器容器,可以按下面的方法在Spring中进行配置:
<bean class="org.springframework.jms.listener.SimpleMessageListenerContainer">
<property name="connectionFactory" ref="connectionFactory" />
<property name="destination" ref="rantzDestination" />
<property name="messageListener" ref="rantzMdp" />
</bean>
对于messageListener属性,我们为其置入了对MDP实现的引用,这样,onMessage()方法将可以被用来接收消息。
使用事务性的MDP:如果收到的一个消息在事务中,则应该使用DefaultMessageListenerContainer:
<bean class="org.springframework.jms.listener.DefaultMessageListenerContainer">
<property name="connectionFactory" ref="connectionFactory" />
<property name="destination" ref="rantzDestination" />
<property name="messageListener" ref="rantzMdp" />
<property name="transactionManager" ref="jmsTransactionManager" />
</bean> 如果事务性需求比较简单,JmsTransactionManager将按如下方法配置:
<bean id="jmsTransactionManager" class="org.springframework.jms.connection.JmsTransactionManager">
<property name="connectionFactory" ref="connectionFactory" />
</bean> 需要提醒的是,tansactionManger属性是可选的。如果不注入事务管理器,MDP就不是事务性的。
编写纯POJO MDP:如果消息监听器容器的messageListener属性被注入了MessageListener的实现,它就能够知道在消息到达时应该调用onMessage()方法。幸运的是,Spring提供了一个替代的MessageListenerAdapter。它是一个MessageListener,可以委派Bean和你选择的方法,见PPT14. 如果不将自己的MessageListener的实现注入到消息监听器容器中,你可以置入到MessageListenerAdapter中:
<bean class="org.springframework.jms.listener.SimpleMessageListenerContainer">
<property name="connectionFactory" ref="connectionFactory" />
<property name="destination" ref="rantzDestination" />
<property name="messageListener" ref ="purePojoMdp" />
</bean> 因为配置了purePojoMdp Bean,所以它是一个MessageListenerAdapter:
<bean id="purePojoMdp" class="org.springframework.jms.listener.adapter.MessageListenerAdapter">
<property name="delegate" ref="rantzMdp" />
<property name="defaultListenerMethod" value="processMotoristInfo" />
</bean> 默认情况下,MessageListenerAdapter在消息到达时会调用handleMessage()方法。但是,我们希望MarketingMdp Bean可以通过 processMotoristInfo()方法处理消息,因此将defaultListenerMethod设置为processMotoristInfo. 因为选择了一个特定的被调用方法,所以不需要实现MessageListener或onMessage()方法。因此 MarketingMdp现在将被简化为PPT15。尽管它是一个POJO,对MapMessage的依赖造成了MarketingMdp与JMS的不必要耦合,另外,MapMessage的getString方法还会抛出必须被处理的JMSException。理想情况下,MarketingMdp不应该依赖任何特定框架的类型。 当MessageListenerAdapter接收消息时,它会考虑消息的类型和defaultListenerMethod的值,并且尝试着查找用来调用的监听器方法符号。PPT16描述了MessageListenerAdapter是如何将JMS消息映射到监听器方法参数的。
转换MDP消息:在最新版中,processMotoristInfo()带有的是Map,并且在处理前需要将Map转换为SpammedMotorist。如果在消息到达时,能够直接给processMotoristInfo()方法传递可以处理的SpammedMotorist对象岂不是更好么?Spring消息转换器可以执行消息和特定域Java类型之间的相互转换工作。在PPT10中,已经有一个消息转换器。我们需要做的就是让MessageListenerAdapter能够感知这个消息转换器。MessageListenerAdapter的messageConverter属性可以完成这项工作:
<bean id="purePojoMdp" class="org.springframework.jms.listener.adapter.MessageListenerAdapter">
<property name="delegate" ref="rantzMdp" />
<property name="defaultListenerMethod" value="processMotoristInfo" />
<propert name="messageConverter" ref="motoristConverter" />
</bean> 现在,可以编写最终版的MarketingMdp了。
使用基于消息的RPC:RPC编程模型可以使与远程服务的交互就如同在调用本地对象的方法。是否有某种方法既具有RPC编程模型的简单性,又可以利用异步消息传递的优点呢?
引入Lingo: 它是一种基于Spring的远程调用方法,它在RPC和异步消息传递之间架起了一座桥梁。与使用其他的Spring远程调用方法相同,Lingo提供了一个服务输出器,可以将Bean的功能输出为Lingo服务和客户端代理。 Lingo远程调用会通过JMS队列或主题来承载信息。 尽管Lingo是基于Spring远程调用的,但是它并不是Spring框架的一部分。你可以从Lingo的主页下载Lingo(http://lingo.codehaus.org/Download).
输出服务:在服务端,Lingo提供了一个服务输出器JmsServiceExporter。被Lingo输出的服务是通过JMS来提供服务的,而不是用于直接的RPC访问,见PPT17。下面的XML配置了一个JmsServiceExporter,能够将rantzMdp Bean输出为RPC-over-JMS服务:
<bean id="server" class="org.logicblaze.lingo.jms.JmsServiceExporter">
<property name="connectionFactory" ref="connectionFactory" />
<property name="destination" ref="destination" />
<property name="service" ref="rantzMdp" />
<property name="serviceInterface" value="com.roadrantz.marketing.MarketingService" />
</bean>
service属性被置入了对rantzMdp Bean的引用,即MarketingMdp。最后,应该将定义了服务的接口的类名配置到serviceInterface属性中。我们声明的被输出服务带有MarketingService接口,定义如下:
public interface MarketingService {
void processMotoristInfo (SpammedMotorist motorist );
} 因为我们定义的服务带有MarketingService接口,这就意味着我们应该对MarketingMdp类进行一些小的修改,以便它可以实现MarketingService接口:
public class MarketingMdp implements MarketingService {
public void processMotoristInfo(SpammedMotorist motorist) {
...
}
}
这就是使用Lingo输出服务的全过程。一旦应用程序启动,JmsServiceExporter将会起作用,就可以开始使用它了。现在,让我们再到客户端看看RoadRantz应用程序是如何调用这个被输出的市场服务的。
代理JMS:在RoadRantz应用程序中,每次用户注册并选择愿意接收特定提供商信息时都需要调用processMotoristInfo()方法,因此,我们必须以某种方法将Lingo输出服务的引用置入到RoadRantz应用程序中。
置入JmsProxyFactoryBean:lingo提供了JmsProxyFactoryBean,这是一个代理工厂Bean,可以生成远程Lingo输出服务的代理。如PPT18所示,通过JmsProxyFactoryBean代理的服务是通过JMS目标(队列或主题)访问的,而不是通过TCP/IP。下面声明配置了一个JmsProxyFactoryBean:
<bean id = "marketing" class="org.logicblaze.lingo.jms.JmsProxyFactoryBean">
<property name="connectionFactory" ref="connectionFactory" />
<property name="destination" ref="destination" />
<property name="serviceInterface" value="com.roadrantz.marketing.MarketingService" />
</bean> serviceInterface属性指定了代理需要实现的Java接口。通过这个接口,RoadRantz可以调用processMotoristInfo()方法。 对于配置JmsProxyFactoryBean,最需要注意的事情是不用配置服务的位置信息。这是因为服务的位置并不重要,只需要知道服务在哪里接收"邮件"。事实上,我们还可以使用远程服务的多个实例。如果想建立一个高可靠性的市场服务,可以启动两个或多个实例,让它们都监听相同的目标。每一个实例都能够处理一个请求。其间,客户端并不知道有一个服务池在等待请求进行响应。
进行调用:在JmsProxyFactoryBean被置入后,就可以开始调用远程服务了。我们需要做的是将其置入到RantServiceImpl:
<bean id ="rantService" class="com.roadrantz.service.RantServiceImpl">
<property name="rantDao" ref="rantDao" />
<property name="marketingService" ref="marketing" />
</bean> 接着,我们可以使用它在addMotorist()方法中向市场服务发送SpammedMotorist对象。PPT19展示了为了调用远程市场服务对RantServiceImpl进行的相关修改。 正如你看到的,调用Lingo输出的服务与调用RMI服务、Web服务或在同一个进程中调用其他Bean上的方法是相同的。PPT19中没有任何关于JMS的内容。唯一不同的地方在Spring的配置。通过这种方法,只需要简单的更改Spring的配置就能够在JMS和其他通信机制直接进行切换。
实例:http://blog.csdn.net/pengchua/article/details/4444400
发表评论
-
spring mvc自动绑定数据到对象
2012-11-05 12:33 1702yxb1990 写道 dancewing 写道 ... -
Spring2.5.6+Struts2.1.6+Hibernate3.2升级到Spring3.1.3+Struts2.3.4+Hibernate3.6Final
2012-11-02 19:50 16281. Struts2.3.4相对于Struts2. ... -
Spring事务配置的五种方式
2012-10-03 22:15 1149前段时间对Spring的事务配置做了比较深入的研究,在此之间 ... -
spring相关配置
2012-09-06 09:05 1145<!-- spring编码过滤器 --> ... -
JAXB “Class has two properties of the same name” issue
2012-08-24 16:53 1413@XmlRootElement(name = "Re ... -
spring注解
2012-06-04 17:34 837spring注解,如果没有设置名称,则默认以你所注解的类的名称 ... -
spring mvc 3.x输出xml或者json
2012-03-11 23:37 2854方式一: 参考: [url] http://www.mkyo ... -
Spring 2.5 jar 所有开发包及完整文档及项目开发实例
2011-08-06 18:29 2910Spring 2.5 jar 所有开发包及完整文档及项目开发实 ... -
jms
2011-07-29 21:54 15651 引言 1.1 编写目的 本文作为B2bi项目中开源产品JO ... -
spring jms
2011-07-29 00:41 16101. 概述:Spring提供了一个用于简化JMS API使用的 ... -
jms
2011-07-29 00:06 1071当前,CORBA、DCOM、RMI等RPC中间件技术已广泛应用 ... -
spring定时器用Annotation实现
2011-07-09 15:34 1987通过 注解 来调度任务 1、AnnotationQuartz ... -
java.lang.NoClassDefFoundError: com/sun/mail/util/LineInputStream
2011-07-07 17:41 1899Exception in thread "main& ... -
spring各种邮件发送
2011-07-07 17:14 1235Spring邮件抽象层的主要包为org.springframe ... -
spring sercurity
2011-07-05 11:04 1289http://www.blogjava.net/redhatl ... -
spring事物
2011-07-04 09:19 1447事务 ( acid ) atomic : 原子性 ,有多个 ... -
spring bean scope
2011-07-01 14:39 1108在spring的配置文件中bean的范围控制用scope来控制 ... -
Spring 框架的设计理念与设计模式分析
2011-06-03 15:18 945http://www.ibm.com/developerwor ... -
spring aop
2011-04-03 07:46 1305Spring中代理类的生成 ... -
spring bean自动扫描分析
2011-01-20 15:16 7379Spring为企业级开发提供了轻量而又强大的支持。但是为了使用 ...
相关推荐
SpringJMS是Spring框架的一部分,它提供了一种与Java消息服务(JMS)进行交互的简单方式。在本文中,我们将深入探讨SpringJMS的基本概念、如何与ActiveMQ集成,以及如何通过示例代码理解其工作原理。 1. **Spring...
详细内容: SpringJMS整合ActiveMQ.doc 详细说明文档 apache-activemq-5.8.0-bin.zip ActiveMQ安装包 JMSTest.rar MyEclipse8.5下web工程
Spring JMS Spring JMS 是一个基于 Java 消息服务(JMS)的框架,它提供了一个简洁的方式来使用 JMS API。Spring JMS 框架提供了一个抽象层,简化了 JMS 的使用,使得开发者可以更容易地使用 JMS。 Spring JMS 的...
【Spring JMS】是Spring框架中的一个模块,用于简化Java消息服务(JMS)的使用,使得异步消息处理变得更加简单和灵活。Spring JMS通过提供一个模板类,抽象了JMS API的底层细节,让开发者能够更加专注于消息的处理...
Spring对JMS提供了很好的支持,可以通过JmsTemplate来方便地实现消息服务。本例通过activeMQ服务器模拟了消息的发送与接收。需要注意的是,activeMQ的运行依赖jdk的环境,而且对jdk的版本也有要求,我用的是jdk1.6+...
Spring JMS(Java Message Service)是Spring框架的一部分,专门用于集成JMS消息传递系统,以实现异步通信和解耦应用程序组件。在这个入门级实例中,我们将探讨如何使用Maven、Spring和ActiveMQ来构建一个简单的...
这篇博客“Spring JMS 消息处理-基于JNDI”将深入探讨如何在Spring应用中使用JMS进行消息处理,并利用JNDI(Java Naming and Directory Interface)来查找和配置消息资源。 JMS是Java平台上的一个标准接口,它定义...
在Java中,Spring框架提供了强大而灵活的支持,使得开发人员能够轻松地实现基于Java消息服务(JMS)的网络编程。JMS是一个标准接口,它允许应用程序创建、发送、接收和读取消息,这些消息可以在不同的应用之间传递,...
本人开发的spring jms项目,已经上线近一年了,这里是我项目jms配置文件,使用的是spring jms 整合weblogic jms。如果真的需要,请咨询我,并且附上我上传的这个配置文件,附近中没有带有这个文件,一律不作任何回答...
在本文中,我们将深入探讨Spring框架中的Java消息服务(JMS)支持,特别是不依赖于JNDI(Java Naming and Directory Interface)的方式。Spring JMS为应用程序提供了与消息中间件交互的能力,允许我们构建可扩展且...
【Spring JMS 知识点详解】 Spring JMS 是 Spring 框架的一部分,它提供了一种简单且灵活的方式来处理 Java 消息服务 (JMS)。在面向服务架构 (SOA) 中,异步消息传递是关键组件,特别是在企业级系统间通信,特别是...
在IT行业中,Spring框架是Java应用开发的基石,它提供了丰富的功能来简化应用程序的构建,而JMS(Java Message Service)则是处理异步通信的一种标准API。Tomcat作为轻量级的应用服务器,常用于部署Spring应用。在这...
**JMS整合Spring实例** Java消息服务(Java Message Service,简称JMS)是Java平台中用于企业级应用间异步通信的一种标准接口。它允许应用程序创建、发送、接收和读取消息,以此来解耦生产者和消费者。而Spring框架...
本项目"jms整合spring工程"是一个已经准备就绪的Java工程,它展示了如何在Spring框架中集成JMS,以便利用消息队列进行通信。主要依赖的是Apache ActiveMQ,这是一款流行的开源JMS提供者,能够实现高效、可靠的实时...
**Spring JMS源码分析** 在Java世界里,消息队列(JMS,Java Message Service)是一种标准,它定义了一种规范,使得不同的消息中间件提供商可以为开发者提供一致性的API,以实现在应用程序间传递消息。Spring JMS是...
在实际应用中,`SpringJMS`可能包含以下示例代码片段: ```xml <bean id="jmsConnectionFactory" class="org.springframework.jms.connection.CachingConnectionFactory"> <!-- 这里配置具体的JMS提供商实现 -->...