web-dev-qa-db-ja.com

Spring BootJMSを使用してトピックを聞く方法

以下のスニペットを使用してトピックを聴こうとしています。ただし、デフォルトではキューをリッスンしています。この場合、xml構成はありません。私は完全に注釈に依存しています。さらに、Springブートによって提供される自動構成に完全に依存しました。 JmsListenerで、宛先タイプをトピックとして設定する方法がわかりません。 SpringJMSの達人が助けてください。

    @Component
    public class MyTopicListener {

        @JmsListener(destination = "${trans.alert.topic}")
        public void receiveMessage(TransactionAlert alert) {
            logger.info("AlertSubscriberEmail :: Sending Email => <" + alert + ">");
        }
    }
6
zikzakjack

完全なSpringブートの例を以下から取得しました: https://github.com/spring-guides/gs-messaging-jms/

これでは、キューからのメッセージの送受信用に作成されます。これをトピックに変更するには、FactoryインスタンスでPub-Subプロパティを設定する必要があります。

import org.springframework.beans.factory.support.BeanDefinitionBuilder;
import org.springframework.beans.factory.support.DefaultListableBeanFactory;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.boot.autoconfigure.jms.DefaultJmsListenerContainerFactoryConfigurer;
import org.springframework.context.ConfigurableApplicationContext;
import org.springframework.context.annotation.Bean;
import org.springframework.jms.annotation.EnableJms;
import org.springframework.jms.config.DefaultJmsListenerContainerFactory;
import org.springframework.jms.config.JmsListenerContainerFactory;
import org.springframework.jms.connection.CachingConnectionFactory;
import org.springframework.jms.core.JmsTemplate;
import org.springframework.jms.support.converter.MappingJackson2MessageConverter;
import org.springframework.jms.support.converter.MessageConverter;
import org.springframework.jms.support.converter.MessageType;

import javax.jms.ConnectionFactory;

@SpringBootApplication
@EnableJms
public class JmsSampleApplication {

public void registerBeans(ConfigurableApplicationContext context ){
    BeanDefinitionBuilder builder = BeanDefinitionBuilder.rootBeanDefinition(JmsTemplate.class);
    CachingConnectionFactory cachingConnectionFactory = new CachingConnectionFactory();

    builder.addPropertyValue("connectionFactory", cachingConnectionFactory);      // set property value
    DefaultListableBeanFactory factory = (DefaultListableBeanFactory) context.getAutowireCapableBeanFactory();
    factory.registerBeanDefinition("jmsTemplateName", builder.getBeanDefinition());
}

@Bean
public JmsListenerContainerFactory<?> topicListenerFactory(ConnectionFactory connectionFactory,
                                                DefaultJmsListenerContainerFactoryConfigurer configurer) {
    DefaultJmsListenerContainerFactory factory = new DefaultJmsListenerContainerFactory();
    factory.setPubSubDomain(true);
    // This provides all boot's default to this factory, including the message converter
    configurer.configure(factory, connectionFactory);
    // You could still override some of Boot's default if necessary.
    return factory;
}

@Bean
public JmsListenerContainerFactory<?> queueListenerFactory(ConnectionFactory connectionFactory,
                                                           DefaultJmsListenerContainerFactoryConfigurer configurer) {
    DefaultJmsListenerContainerFactory factory = new DefaultJmsListenerContainerFactory();
    //factory.setPubSubDomain(true);
    // This provides all boot's default to this factory, including the message converter
    configurer.configure(factory, connectionFactory);
    return factory;
}

@Bean // Serialize message content to json using TextMessage
public MessageConverter jacksonJmsMessageConverter() {
    MappingJackson2MessageConverter converter = new MappingJackson2MessageConverter();
    converter.setTargetType(MessageType.TEXT);
    converter.setTypeIdPropertyName("_type");
    return converter;
}
public static void main(String[] args) {
    ConfigurableApplicationContext context = SpringApplication.run(JmsSampleApplication.class, args);

    JmsTemplate jmsTemplate = context.getBean(JmsTemplate.class);

    // Send a message with a POJO - the template reuse the message converter
    System.out.println("Sending an email message.");
    jmsTemplate.convertAndSend("mailbox.topic", new Email("[email protected]", "Hello"));
    jmsTemplate.convertAndSend("mailbox.queue", new Email("[email protected]", "Hello"));

    }
}

リスナー

package org.springboot.jms;

import org.springframework.jms.annotation.JmsListener;
import org.springframework.stereotype.Component;

/**
 * Created by RGOVIND on 10/20/2016.
 */
@Component
public class HelloTopicListener {

    @JmsListener(destination = "mailbox.topic", containerFactory = "topicListenerFactory")
    public void receiveTopicMessage(Email email) {
        System.out.println("Received <" + email + ">");
    }

    @JmsListener(destination = "mailbox.queue", containerFactory = "queueListenerFactory")
    public void receiveQueueMessage(Email email) {
        System.out.println("Received <" + email + ">");
    }
}

これが完了すると、選択したトピックにサブスクライブする準備が整います。

もちろん、これには複数のアプローチがあります。さまざまなjmsTemplatesのBe​​anのマップを作成できます。それぞれのマップは、キューまたはトピックに基づいて必要なときに使用できます。テンプレートとBeanは、これで説明されているように選択した方法でインスタンス化できます SO質問 。それが役に立てば幸い

12

正解とマークされた答えはほとんど正解です。それでも機能しません:

factory.setPubSubDomain(true) 

後に来る必要があります:

configurer.configure(factory, connectionFactory);

そうしないと、デフォルトを構成するときにtrueに設定されているpubSubDomainフラグが失われ、そのファクトリインスタンスはトピックではなくキューで引き続き機能します。

14
dleerob

Spring BootのApplication.propertiesで、次のプロパティを設定してみてください。

spring.jms.pub-sub-domain=true

次に、トピックをリッスンするために使用しているコンテナファクトリにこのプロパティを使用します。

5
Aditya K