`

Spring JMS 消息处理-基于JNDI

阅读更多

        Spring JMS 把企业消息处理变得轻而易举,本文将使你快速了解使用Spring JMS框架和IBM WebSphere MQ7.5.0.2进行JMS消息处理的基础知识。

        JMS PG 定义了 Java 应用程序通过面向消息的中间件(MOM)创建和交换消息的标准途径,下面将使用一个简单的示例来演示 Spring JMS 的特性。您将随我一道开发一个点对点的(P2P)基于消息的系统,使用 Spring JMS 框架通过 JMS 接口与 IBM 的 WebSphere MQ 集成。完成练习后,将可以通过这个系统发送和接收简单的文本消息。

一.Spring JMS

        Spring 的 JMS 抽象框架简化了 JMS API 的使用,并与 JMS 提供者(比如 IBM 的 WebSphere MQ 5.3)平滑地集成。org.springframework.jms.core 包提供了在 Spring 中使用 JMS 的核心功能。它的模板类处理资源的创建和释放,简化了 JMS 的使用。

        像其他大多数 Spring 模板类一样,JMS 模板类提供了执行公共操作的 helper 方法。在需要更复杂应用的情况下,类把处理任务的核心委托给用户实现的回调接口。JMS 类提供了方便的方法,用来发送消息、同步地使用消息以及向用户公开 JMS 会话和消息的制作者。

        以下 JMS 包和 org.springframework.jms.core 一起构成了 Spring JMS 的功能:

org.springframework.jms.support

        提供转换 JMSException 的功能。转换代码把检测到的 JMSException 层次结构转换成未检测到异常的镜像层次结构。

org.springframework.jms.support.converter

        提供 MessageConverter 抽象,以在 Java 对象和 JMS 消息之间进行转换。

org.springframework.jms.support.destination

        提供管理 JMS 目标的不同策略,比如针对 JNDI 中保存的目标的服务定位器。

org.springframework.jms.connection

        提供适合在独立应用程序中使用的 ConnectionFactory 实现。connection 还包含针对 JMS 的 Spring PlatformTransactionManager 实现。它允许把 JMS 作为事务性资源集成到 Spring 的事务管理机制中。

 

二.IBM WebSphere MQ

        就像前面提到的,示例应用程序会用 Spring 的 JMS 框架通过 JMS 接口与 IBM 的 WebSphere MQ 集成。通过在应用程序和 Web 服务之间传递消息,WebSphere MQ 提供了可靠的、有恢复能力的应用程序集成。它使用队列和事务性工具帮助保持消息跨网络的完整性。WebSphere MQ 降低了信息丢失的风险和调和通信 IT 系统的需要。

        WebSphere MQ 在它所支持的所有平台上提供了一致的应用程序编程接口,这有助于让集成的程序可移植。除了标准接口外,WebSphere MQ 还完整实现了JMS 接口,包括对发布-订阅消息传递的支持。WebSphere MQ Explorer 工具可以远程地管理和配置整个 MQ 网络。管理和配置工具基于开放源码的 Eclipse 框架,而且是可扩展的。

 

三.Spring JMS 模板

        Spring 框架提供了 JmsTemplate 的两个实现。JmsTemplate 类使用 JMS 1.1 API,子类 JmsTemplate102 则使用 JMS 1.0.2 API。我的示例应用程序使用的是 JmsTemplate102。

        JMS 模板被用来发送和接收 JMS 消息。Spring 采用回调机制对 JMS 信息传递进行协调。MessageCreator 回调接口用 JmsTemplate 中的调用代码提供的 Session 创建消息。为了支持 JMS API 更复杂的应用,回调 SessionCallback 向用户提供了 JMS 会话,而 callback ProducerCallback 则公开了 Session 和 MessageProducer 组合。

        下面显示了示例应用程序使用的 JMS 模板的配置。

<!-- JMS Queue Template -->
<bean id="jmsQueueTemplate" 
	  class="org.springframework.jms.core.JmsTemplate102">
<property name="connectionFactory">
  <ref bean="jmsQueueConnectionFactory"/>
</property>
<property name="destinationResolver">
  <ref bean="jmsDestinationResolver"/>
</property>
<property name="pubSubDomain">
  <value>false</value>
</property>
<property name="receiveTimeout">
  <value>20000</value>
</property>
</bean>

        jmsQueueTemplate bean 与 JMS 连接工厂和 JMS 目标解析器绑定在一起,用于解析 JMS 客户机通过 JNDI 提供的目标队列名。connectionFactory 属性指定了如何获得到 JMS 提供者的连接。如下显示了如何从 JNDI 检索连接工厂。

        通过 JNDI 配置 JMS 连接工厂

<!-- JMS Queue Connection Factory -->
<bean id="internalJmsQueueConnectionFactory"
	  class="org.springframework.jndi.JndiObjectFactoryBean">
    <property name="jndiTemplate">
          <ref bean="jndiTemplate"/>
    </property>
    <property name="jndiName">
          <value>MQ_JMS_MANAGER</value>
    </property>
</bean>

        可以看到,JndiObjectFactoryBean 被绑定到 internalJmsQueueConnectionFactory。JndiObjectFactoryBean 用 JndiTemplate 属性进行 JNDI 查询。Spring 将用 JndiTemplate 中指定的环境属性和初始上下文在 JNDI 中查询连接工厂。如下显示了 JndiTemplate 配置 bean 的配置。

        JNDI 查询的 JNDI 模板配置

<bean id="jndiTemplate" class="org.springframework.jndi.JndiTemplate">
	<property name="environment">
	  <props>
		<prop key="java.naming.factory.initial">
			com.sun.jndi.fscontext.RefFSContextFactory
		</prop>
		<prop key="java.naming.provider.url">
			file:/C:/JNDI-Directory
		</prop>
	  </props>
	</property>
</bean>

        以上配置进行 JNDI 查询时用 com.sun.jndi.fscontext.RefFSContextFactory 指定初始上下文工厂,用基于文件的 file:/C:/JNDI-Directory 作为提供者 URL。根据示例应用程序的意图,JNDI 访问会采用基于文件的 FSContext 版本的配置把 MQ 队列绑定到 JNDI。

        有了定义好的 JMS 模板,下一步就是把它绑定到示例应用程序中,然后就可以用它发送和接收消息了。

 

四.Spring JMS 实现

        JMS 模板可以绑定到应用程序中,以发送和接收 JMS 消息。如下可以看出如何把JMS模板绑定到示例应用程序中。

        把 JmsTemplate 绑定到应用程序中

<bean id="jmsSender" 
	  class="springexample.client.JMSSender">
	<property name="jmsTemplate102">
	  <ref bean="jmsQueueTemplate"/>
	</property>
</bean>
<bean id="jmsReceiver" 
	  class="springexample.client.JMSReceiver">
	<property name="jmsTemplate102">
	<ref bean="jmsQueueTemplate"/>
  </property>
</bean>

        可以看到,我把 jmsQueueTemplate 绑定到用来发送和接收消息的 JmsSender 应用程序 bean 和 JmsReceiver bean。如下显示了与 JMSSender 类有关的代码。

        用JmsTemplate 发送 JMS 消息的 JMSSender

public class JMSSender {
   private JmsTemplate102 jmsTemplate102;
   public JmsTemplate102 getJmsTemplate102() {
     return jmsTemplate102;
   }
   public void setJmsTemplate102(JmsTemplate102 jmsTemplate102) {
     this.jmsTemplate102 = jmsTemplate102;
   }
   public void sendMesage(){
     jmsTemplate102.send("JMS_RequestResponseQueue", 
                  new MessageCreator() {
        public Message createMessage(Session session) 
                  throws JMSException {
          return session.createTextMessage("This is a sample message");
        }
      });
   }

        JMSSender 类用 jmsTemplate102.send() 方法发送 JMS 消息。send() 方法的第一个参数是 JNDI 队列名,队列名指定了消息应当发送到哪里。(很快就会看到如何把 WebSphere MQ 的队列名绑定到 JNDI。)send() 方法的第二个参数是 MessageCreator 类。JmsTemplate 中的调用代码提供了 Session 类,这个类提供了一个创建 JMS 消息的回调接口。

        下一步是用 JMS 的 Session 类创建一个简单的文本消息。在代码执行时,消息会传递给 WebSphere MQ 服务器的队列。如下显示了使用 JmsTemplate 检索 JMS 消息的 JMSReceiver 应用程序 bean 的代码。

        用JmsTemplate 检索 JMS 消息的 JMSReceiver

public class JMSReceiver {
    private JmsTemplate102 jmsTemplate102;
    public JmsTemplate102 getJmsTemplate102() {
      return jmsTemplate102;
    }
    public void setJmsTemplate102(JmsTemplate102 jmsTemplate102) {
     this.jmsTemplate102 = jmsTemplate102;
    }
    public void processMessage(){
      Message msg = jmsTemplate102.receive("JMS_RequestResponseQueue");
      try{
        TextMessage textMessage = (TextMessage) msg;
        if( msg!=null){
        System.out.println(" Message Received -->" + 
                    textMessage.getText());
        }
      }catch(Exception e){
            e.printStackTrace();
      }
    }
}

        JMSReceiver 类用 jmsTemplate102.receive() 方法同步地接收 JMS 消息。receive() 方法指定 JNDI 队列名,并从中检索消息。JMSTemplate 类的 processMessage() 方法由接收 JMS 客户机调用。JSMTemplate bean 的属性 receiveTimeout(列在 JMSTemplate 配置中)指定接收客户机同步地从队列中接收消息时要等候的时间。

        现在应用程序的代码已完成!下一步就是配置 WebSphere MQ 队列并把它们绑定到 JNDI 对象。

 

五.队列管理器的设置

        在运行应用程序之前,需要设置 WebSphere MQ 的队列管理器和队列,并把它们绑定到 JNDI。如果喜欢的话,可以按照这部分的示例做:只需将下载的设置 WebSphere MQ 队列的批文件和应用程序的源代码和部署描述符即可。

1.设置队列

        运行SpringSeriesPart4JMS\batch文件夹中的 mqsetup.bat 文件。这个批文件要求在 path 环境变量中设置好 MQ 安装的 bin 文件夹(例如IBM\WebSphere MQ\bin)。运行了批文件之后,应当看到消息 “All valid MQSC commands were processed”。要打开 MQ Explorer 并检查已经创建的队列管理器和队列,请选择 Start -> Programs -> IBM WebSphere MQ -> WebSphere MQ Explorer。下图显示出示例应用程序 QueueManagerMQJMS.QManager 已经创建并正在运行。


        这就完成了队列的设置。

2.设置 JMS 和 JNDI 管理

        在示例应用程序中,JNDI 的访问利用了可以从 JNDI 主页得到的基于文件的 FSContext 版本。FSContext.jar 文件也包含在 WebSphere MQ 的 JMS 支持当中。请添加文件夹 \MQSeriesInstallable\WebSphere MQ\Java\lib 和 \MQSeriesInstallable\WebSphere MQ\Java\bin 到系统的 PATH 环境变量中。而且,请把 \MQSeriesInstallable\WebSphere MQ\Java\lib 文件夹中的所有 jar 文件添加到系统的 CLASSPATH 环境变量中。还可以运行 C:\SpringSeriesPart4JMS\batch 文件夹中的 classpath.cmd 文件,它会设置必要的 path 和 CLASSPATH 变量。要做到这点,只需要修改 classpath.cmd 文件中的 MQ_JAVA_INSTALL_PATH,把它指到 WebSphere MQ JMS 的安装目录。

        接下来,修改 \MQSeriesInstallableDirectory\WebSphere MQ\Java\bin 中的JMSAdmin.config 配置文件,WebSphere MQ JMS 管理程序用它指明应用程序要使用的上下文工厂和 JNDI 实现的地址。请取消以下行的注释:

INITIAL_CONTEXT_FACTORY=com.sun.jndi.fscontext.RefFSContextFactory

        并注释掉其余两个 INITIAL_CONTEXT_FACTORY 变量。还要取消以下行的注释:

PROVIDER_URL=file:/C:/JNDI-Directory

        并注释掉其余两个 PROVIDER_URL 变量。

        可以在\SpringSeriesPart4JMS\batch 文件夹中发现参考的示例配置文件。

        为了保存 JNDI 对象,请在驱动器 C:上创建名为 JNDI-Directory 的目录。切换到 \MQSeriesInstallableDirectory\WebSphere MQ\Java\bin目录并运行JMSAdmin批文件,应当看到 InitCtx 变量。逐个输入以下内容:

def qcf(MQ_JMS_MANAGER) qmgr(MQJMS.QManager)

        按回车

def q(JMS_RequestResponseQueue) qmgr(MQJMS.QManager)  queue(RequestResponseQueue)
        按回车


        现在已经把 WebSphere MQ 队列绑定到 JNDI 对象,作为应用程序客户可以通过 JNDI 查询对象。现在剩下的就是看代码的实际作用了!

 

六.运行示例

        要运行示例,请下载Spring框架和它的所有依赖文件并解压,请把 Spring 库和 commons-logging.jar拷贝到 SpringSeriesPart4JMS\lib 文件夹。还要把所有的 jar 库从 \MQSeriesInstallableDirectory\WebSphere MQ\Java\lib 目录拷贝到 SpringSeriesPart4JMS\lib 文件夹。其中包含 MQseries 和 JMS 的相关库。

        将SpringSeriesPart4JMS引入到Eclipse中并运行 SendMQSpringJMS 类,它会调用 JMSSender 类,发送消息到WebSphere MQ RequestResponse队列。SendMQSpringJMS还会通过它的ClassPathXmlApplicationContext装入 spring 配置文件。一旦 bean 全部装载,就可以通过 Spring 的 ApplicationContext 的 getBean() 方法访问 JMSSender。

        装入示例应用程序的 Spring 配置

ClassPathXmlApplicationContext appContext = 
new ClassPathXmlApplicationContext(new String[] { "spring/spring-mqseries-jms.xml"
});
JMSSender jmsSender = (JMSSender)appContext.getBean("jmsSender");
        运行SendMQSpringJMS类,在控制台上会打印出发以下消息:
SendMQSpringJMS started
Classpath loaded
SendMQSpringJMS end

        MQ队列管理器中将会多一条消息,如下所示。


        消息传递到队列上之后,请运行 JMS 接收方客户机以检索消息。运行 ReceiveMQSpringJMS 类,该类会调用 JMSReceiver 类,以从 WebSphere MQ 的 RequestResponse 队列接收文本消息。在控制台上会打印出以下消息:

ReceiveMQSpringJMS started
Classpath loaded
 Message Received -->This is a sample message
ReceiveMQSpringJMS end

 

七.结束语

        在Spring 系列的这篇文章中,我们学习了 Spring JMS 框架的基础。首先介绍了示例应用程序的核心组件——Spring JMS 框架和 IBM 的 WebSphere MQ7.5.0.2,然后介绍了如何用 Spring JMS 模板向 WebSphere MQ 队列发送消息和从中接收消息。虽然这个示例非常简单,但是可以把这里介绍的步骤应用到更复杂的应用程序。

 

文章来源:http://www.ibm.com/developerworks/cn/java/wa-spring4/

  • 大小: 57.1 KB
  • 大小: 52.5 KB
  • 大小: 19.8 KB
分享到:
评论

相关推荐

    Spring JMS消息处理-不基于JNDI

    本篇将主要探讨在不依赖于JNDI(Java Naming and Directory Interface)的情况下,如何使用Spring JMS处理消息。 JNDI通常被用于查找和绑定分布式环境中的资源,如消息队列。然而,在某些场景下,我们可能不需要...

    spring-jms-oracle-aq.rar_oracle aq_spring oracle aq_spring oracl

    如果在处理消息时发生异常,整个事务将被回滚,消息将保留在队列中,等待重新处理。 6. 性能优化: 根据应用程序的需求,你可以调整MessageListenerContainer的属性,如并发消费者数量、接收超时等,以优化性能和...

    Spring-JMS把企业消息处理变容易.doc

    Spring JMS 是一个强大的框架,它极大地简化了Java企业级消息处理。它通过提供一套抽象和模板类,使得开发者能够更加便捷地使用Java消息服务(JMS),并与各种JMS提供者,如IBM的WebSphere MQ进行集成。本文将深入...

    Spring发送接收JMS消息

    **Spring与JMS消息传递** 在Java世界中,Java Message Service (JMS) 是一个标准接口,用于在分布式环境中发送和接收消息。Spring框架提供了一种简单而强大的方式来集成JMS,使得开发者可以轻松地在应用中实现异步...

    spring-jms-4.2.xsd.zip

    例如,`&lt;jee:jndi-lookup&gt;`用于查找JMS连接工厂,`&lt;jms:listener-container&gt;`定义一个监听器容器,`&lt;jms:listener&gt;`定义消息监听器等。 总之,"spring-jms-4.2.xsd.zip"是一个包含了Spring JMS 4.2版本的XML模式...

    spring-framework-3.0.0.M4-with-docs

    org.springframework.jms-3.0.0.M4.jar: 为简化JMS API的使用而作的简单封装 org.springframework.beans-3.0.0.M4.jar: SpringIoC的基础实现 org.springframework.core-3.0.0.M4.jar: 提供的基础核心功能 org....

    Spring+weblogic接收JMS消息

    - `MessageListenerContainer`配置了消息监听器,当接收到消息时自动调用监听器的方法处理消息。 - `MessageConverter`转换消息对象,如将Java对象转换为JMS消息格式。 2. **WebLogic Server中的JMS服务**: - ...

    Spring JMS使异步消息变得简单.doc

    通过对比传统JMS实现和Spring JMS,我们可以看到Spring JMS显著减少了开发者处理消息时的代码量和复杂性,使得异步消息处理的实现更为简洁。这不仅降低了出错的可能性,也使得代码更易于测试和维护。Spring JMS的...

    spring-framework-4.3.30.RELEASE-schema.zip

    "jms"(Java Message Service)模块是Spring对消息中间件的支持,用于实现异步通信和解耦。通过定义消息生产者和消费者,Spring JMS提供了一种在分布式系统中可靠传递信息的方式,增强了系统的可扩展性和可靠性。 ...

    Spring JMS

    Spring JMS 是一个基于 Java 消息服务(JMS)的框架,它提供了一个简洁的方式来使用 JMS API。Spring JMS 框架提供了一个抽象层,简化了 JMS 的使用,使得开发者可以更容易地使用 JMS。 Spring JMS 的特点是提供了...

    spring,weblogic配置jms

    &lt;jee:jndi-lookup id="jmsConnectionFactory" jndi-name="jms/MyConnectionFactory" resource-ref="true"/&gt; &lt;jee:jndi-lookup id="myQueue" jndi-name="jms/MyQueue" resource-ref="true"/&gt; ``` 4. **配置Spring的...

    spring jms tomcat 异步消息传递入门实例

    4. **创建消息消费者**:为了接收消息,你需要创建一个`MessageListener`实现,如`MessageListenerAdapter`,它将监听指定的目的地,并调用你提供的方法来处理消息。在Spring配置中,将监听器绑定到目标目的地。 5....

    spring整合JMS-居于ActiveMQ实现

    Spring的`MessageListenerAdapter`使得监听器的实现变得简单,只需要提供处理消息的方法即可。例如: ```java @Component public class MyMessageListener { @JmsListener(destination = "myQueue") public void ...

    使用Spring JMS轻松实现异步消息传递.pdf

    Spring JMS 是 Spring 框架的一部分,它提供了一种简单且灵活的方式来处理 Java 消息服务 (JMS)。在面向服务架构 (SOA) 中,异步消息传递是关键组件,特别是在企业级系统间通信,特别是与外部系统交互时。Java 消息...

    spring-jms使用queue发送消息简单例子

    在Spring框架中,Spring JMS(Java Message Service)是一个用于集成JMS的模块,它提供了一个抽象层,简化了与消息中间件的交互。在这个"spring-jms使用queue发送消息简单例子"中,我们将深入探讨如何使用Spring JMS...

Global site tag (gtag.js) - Google Analytics