3

Github

在上一篇文章使用Spring/Spring Boot集成JMS的陷阱中讲到了在使用Spring JMS组件时存在这一些性能陷阱,本文会着该文讲解一下Spring JMS的各个组件、它们的作用以及正确使用的方法。

JmsTemplate

用来send、receive消息(receive是同步式的,会block)。每个JmsTemplate实例拥有自己的配置,比如:connectionFactorysessionTransactedsessionAcknowledgeModedeliveryModetimeToLive等等,所以需根据不同场景配置提供不同的JmsTemplate Bean而不是一个Singleton Bean通吃所有JMS操作。

下面是类图(只包含了部分关键属性):

JmsTemplate

ConnectionFactory

Spring提供了两个javax.jms.ConnectionFactory的实现:SingleConnectionFactoryCachingConnectionFactory。它们实际上是一种Wrapper,用来缓存如:ConnectionSessionMessageProducerMessageConsumer

事实上JmsTemplate的Javadoc提到过:

NOTE: The ConnectionFactory used with this template should return pooled Connections (or a single shared Connection) as well as pooled Sessions and MessageProducers. Otherwise, performance of ad-hoc JMS operations is going to suffer.

在Spring JMS文档的Caching Messaging Resources中也提到了需要优化资源使用以提升性能:

The standard API involves creating many intermediate objects. To send a message the following 'API' walk is performed
ConnectionFactory->Connection->Session->MessageProducer->send
Between the ConnectionFactory and the Send operation there are three intermediate objects that are created and destroyed. To optimise the resource usage and increase performance two implementations of ConnectionFactory are provided.

下面是类图(只包含了部分关键属性):

ConnectionFactory

SingleConnectionFactory

SingleConnectionFactory顾名思义,无论调用多少次createConnection(..)都返回同一个Connection实例。但是它并不缓存Session,也就是说调用一次createSession(...)就会创建一个新的实例。

可以通过SingleConnectionFactoryTest了解详情。

所以在大多数情况下,不推荐使用SingleConnectionFactory

CachingConnectionFactory

CachingConnectionFactory继承自SingleConnectionFactory,除了依然保留缓存同一个Connection实例的特性外,还增加了对于SessionMessageProducerMessageConsumer的缓存。

CachingConnectionFactory其内部维护了一个Acknowledge Mode -> List<Session>的Map,sessionCacheSize实际上指的是List<Session>的大小,所以最多会有4 * sessionCacheSize数量的Session被缓存(因为JMS规定了四种Acknowledge Mode)。
并且CachingConnectionFactory其本质不是一个Object Pool,所以不会因为实际请求Session数量超出sessionCacheSize导致block或者返回null,可以放心使用。

CachingConnectionFactory返回的每个Session内部都有ConsumerCacheKey -> MessageConsumer以及DestinationCacheKey -> MessageProducer的Map,用来缓存MessageProducerMessageConsumer

可以通过CachingConnectionFactory了解详情。

MessageListenerContainer

Spring JMS中有一个特性MessageListenerContainer,按照官方文档的说法:

A message listener container is used to receive messages from a JMS message queue and drive the MessageListener that is injected into it.

上面提到的MessageListener就是javax.jms.MessageListener,第一次看到这个东西感觉有点奇怪,因为MessageListener的正规用法应该MessageConsumer.setMessageListener()就行了。

因为MessageListenerContainer继承自SmartLifeCycle,所以它提供了程序启动时开启connection、session,程序关闭是关闭session、connection的功能,能够让你不用操心资源回收问题。

下面介绍一下两个实现SimpleMessageListenerContainerDefaultMessageListenerContainer

下面是类图(只包含了部分关键属性):

MessageListenerContainer

SimpleMessageListenerContainer

SimpleMessageListenerContainer使用MessageConsumer.setMessageListener()来监听消息,它不支持参与外部事务(比如PlatformTransactionManager)。

它是可以持有多个MessageConsumer实例的。代码如下:

// ...

private int concurrentConsumers = 1;
private Set<Session> sessions;
private Set<MessageConsumer> consumers;

// ...
protected void initializeConsumers() throws JMSException {
  // Register Sessions and MessageConsumers.
  synchronized (this.consumersMonitor) {
    if (this.consumers == null) {
      this.sessions = new HashSet<Session>(this.concurrentConsumers);
      this.consumers = new HashSet<MessageConsumer>(this.concurrentConsumers);
      Connection con = getSharedConnection();
      for (int i = 0; i < this.concurrentConsumers; i++) {
        Session session = createSession(con);
        MessageConsumer consumer = createListenerConsumer(session);
        this.sessions.add(session);
        this.consumers.add(consumer);
      }
    }
  }
}

其处理消息的方式有两种:1)传统的MessageConsumer.setMessageListener();2)使用Executor

if (this.taskExecutor != null) {
  consumer.setMessageListener(new MessageListener() {
    @Override
    public void onMessage(final Message message) {
      taskExecutor.execute(new Runnable() {
        @Override
        public void run() {
          processMessage(message, session);
        }
      });
    }
  });
}
else {
  consumer.setMessageListener(new MessageListener() {
    @Override
    public void onMessage(Message message) {
      processMessage(message, session);
    }
  });
}

DefaultMessageListenerContainer

DefaultMessageListenerContainerSimpleMessageListenerContainer不同,它使用MessageConsumer.receive()来处理消息,并且支持XA transaction。

因为receive()是同步的、blocking方法,其性能没有setMessageListener()好,所以它非常依赖多线程(TaskExecutor),这也也带来来dynamic scaling的好处。

请注意不要对Topic采用多线程,否则会收到重复的消息,详情见官方文档

异步接收消息

同步接收消息的方式有JmsTemplate.receive*()MessageConsumer.receive*(),这里不多讲,重点讲异步接收消息的几种方式。

MessageListener & MessageListenerContainer

MessageListener包装到MessageListenerContainer里接收消息,例子参见官方文档Asynchronous Reception - Message-Driven POJOs

SessionAwareMessageListener

SessionAwareMessageListener是Spring提供和MessageListener类似的接口,MessageListenerContainer支持这个接口,用法和MessageListener一样。

MessageListenerAdapter

MessageListenerAdapter是Spring提供的另一个异步接收消息的方式,它MessageListenerSessionAwareMessageListener更灵活,因为它采用反射机制来把消息传递到你的接收消息的方法上。

使用方法见官方文档the SessionAwareMessageListener interface

@JmsListener

@JmsListener是另一种接收消息的方法,怎么使用它可以看官方文档Annotation-driven listener endpoints

@JmsListenerMessageListenerSessionAwareMessageListenerMessageListenerAdapter一样也需要一个Container,用户可以通过@JmsListener.containerFactory属性来指定JmsListenerContainerFactory

Spring提供了两种JmsListenerContainerFactory实现:

  1. DefaultJmsListenerContainerFactory,用来生产DefaultMessageListenerContainer,Spring Boot提供DefaultJmsListenerContainerFactoryConfigurer作为配置工具
  2. SimpleJmsListenerContainerFactory,用来生产SimpleMessageListenerContainer

所以在使用@JmsListener需要仔细的选择正确的JmsListenerContainerFactory,而不是全局采用一种配置。

总结

使用Spring JMS时有需要注意以下三点:

  1. 根据实际情况,配置合适的ConnectionFactory Bean,如有需要可以有多个ConnectionFactory Bean。
  2. JmsTemplate, MessageListenerContainer, JmsListenerContainerFactory需根据实际情况配置不同Bean,避免全局使用一套。
  3. JmsTemplate, MessageListenerContainer, JmsListenerContainerFactory选择合适的ConnectionFactory。
  4. 设定好合适的Executor/线程池大小,避免大量Thread block。

下面是一张各个组件的关系图。

关系图

参考资料


chanjarster
4.2k 声望244 粉丝