java - spring jms 全局 ErrorHandler 不工作

标签 java spring jms spring-jms

MyProjectJmsListenerContainerFactory 类:

@Configuration
public class MyProjectJmsListenerContainerFactory {

    @Bean
    public JmsListenerContainerFactory<?> myFactory(
            ConnectionFactory connectionFactory,
            DefaultJmsListenerContainerFactoryConfigurer configurer, 
            MyProjectJmsGlobalErrorHandler errorHandler) {
        ActiveMQConnectionFactory activeMQConnectionFactory =
                       (ActiveMQConnectionFactory)connectionFactory;
        RedeliveryPolicy redeliveryPolicy = new RedeliveryPolicy();
        redeliveryPolicy.setMaximumRedeliveries(1);
        activeMQConnectionFactory.setRedeliveryPolicy(redeliveryPolicy);

        DefaultJmsListenerContainerFactory factory = 
                 new DefaultJmsListenerContainerFactory();
        factory.setErrorHandler(errorHandler);
        configurer.configure(factory, activeMQConnectionFactory);
        return factory;
    }
}

MyProjectJmsGlobalErrorHandler 类:

@Component
public class MyProjectJmsGlobalErrorHandler implements ErrorHandler {

    @Override
    public void handleError(Throwable throwable) {
        System.out.println("JMS EXCEPTION HANDLER CAUGHT THE EXCEPTION !!");       
    }
}

JMSTextMessageConverter 类:

@Configuration
public class JMSTextMessageConverter {

    @Bean
    public MessageConverter jacksonJmsMessageConverter() {
        MappingJackson2MessageConverter mappingJackson2MessageConverter = 
                     new MappingJackson2MessageConverter();
        mappingJackson2MessageConverter.setTargetType(MessageType.TEXT);
        mappingJackson2MessageConverter.setTypeIdPropertyName("_type");

        return mappingJackson2MessageConverter;
    }   
}

JMSMessagePublisher 类:

@Component
public class JMSMessagePublisher {

    private JmsTemplate jmsTemplate;

    private MessageConverter messageConverter;

    @Autowired
    public JMSMessagePublisher(JmsTemplate jmsTemplate,    
                           MessageConverter messageConverter) {
        this.jmsTemplate = jmsTemplate;
        this.messageConverter = messageConverter;
    }

    public void publishMessage() {
        jmsTemplate.setMessageConverter(messageConverter);
        jmsTemplate.convertAndSend("test.q1", "Hello World");
    }
}

JMSListener 类:

@Component
public class JMSListener {

    @JmsListener(destination = "test.q1")
    public void subscribe(String message) throws Exception {
        System.out.println(" message received ::::::::"+message);
        throw new Exception("Exception from message listener ");
    }
}

我从上面的 subscribe 方法抛出异常,但无法收到 ErrorHandler 通知,并且我可以在日志中注意到以下警告。

WARN 1993 --- [enerContainer-1] o.s.j.l.DefaultMessageListenerContainer  : 
Execution of JMS message listener failed, and no ErrorHandler has been set.

我的 JMS 配置有什么问题,即如何设置 ErrorHandler 以便在 handleError(Throwable throwable) 方法内全局捕获/通知所有异常.

我使用 JMSMessagePublisher 类(上面给出)的 publishMessage() 方法发布消息。

最佳答案

ErrorHandler 仅用于处理监听器在处理消息时抛出的错误(异常)。

由于没有连接,因此还没有可以处理错误的消息。

您可以编写自己的逻辑来尝试不时从连接工厂获取连接。

编辑

我刚刚进行了测试,它对我来说效果很好......

@SpringBootApplication
public class So49861714Application {

    public static void main(String[] args) {
        SpringApplication.run(So49861714Application.class, args);
    }

    @Bean
    public ApplicationRunner runner(JmsTemplate template) {
        return args -> template.convertAndSend("foo", "testMessage");
    }

    @Bean
    public DefaultJmsListenerContainerFactory jmsListenerContainerFactory(
            DefaultJmsListenerContainerFactoryConfigurer configurer,
            ConnectionFactory connectionFactory,
            ErrorHandler myErrorHandler) {
        DefaultJmsListenerContainerFactory factory = new DefaultJmsListenerContainerFactory();
        configurer.configure(factory, connectionFactory);
        factory.setErrorHandler(myErrorHandler);
        return factory;
    }

    @Bean
    public ErrorHandler myErrorHandler() {
        return t -> {
            System.out.println("In error handler");
            t.printStackTrace();
        };
    }

    @JmsListener(destination = "foo")
    public void listen(String in) {
        System.out.println(in);
        throw new RuntimeException("test");
    }

}

testMessage
In error handler
org.springframework.jms.listener.adapter.ListenerExecutionFailedException: Listener method 'public void com.example.So49861714Application.listen(java.lang.String)' threw exception; nested exception is java.lang.RuntimeException: test
...
Caused by: java.lang.RuntimeException: test

我刚刚注意到您的工厂 bean 名为 myFactory 但您没有在 @JmsListener 中指定它。 如果您的应用程序是 Spring Boot 应用程序,我猜测监听器正在使用 Boot 的 jmsListenerContainerFactory bean(这是默认 bean 名称)。

关于java - spring jms 全局 ErrorHandler 不工作,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/49861714/

相关文章:

spring - 当 onMessage() 抛出 JMSException 时,JMS 队列会发生什么?

java - 使用JMS实现通知需要什么?

ssl - 如何在通过 JNDI 命名服务器连接时为 Hornetq 客户端指定本地自定义 SSL 信任库

java - 未经授权的错误 : Full authentication is required to access this resource with spring. 安全 v.6

java - Spring Boot JPA - 没有类型的合格bean

java - Java 对于重载方法最具体的方法规则如何与多个参数一起工作?

java - Java中稀疏矩阵行的高效修改

spring - Spring Boot 中配置的 Swagger 仅显示具有 POST 和 GET 映射的方法

java - JSP 不向 JQuery AJAX 返回数据

java - 如何使用 Axis 将 Java 日期传递给 .Net web 服务