注解方法的消息转换

在调用监听器之前,管线中有两个转换步骤。第一步使用 MessageConverter 将输入的 Spring AMQP Message 转换为 Spring-messaging Message。当目标方法被调用时,消息载荷会在必要时转换为方法参数类型。

第一步的默认 MessageConverter 是 Spring AMQP 的 SimpleMessageConverter,它处理向 Stringjava.io.Serializable 对象的转换。所有其他对象仍保持为 byte[]。在下面的讨论中,我们将此称为“消息转换器”。

第二步的默认转换器是 GenericMessageConverter,它委托给一个转换服务(DefaultFormattingConversionService 的一个实例)。在下面的讨论中,我们将此称为“方法参数转换器”。

要更改消息转换器,可以将其作为属性添加到容器工厂 bean 中。下面的示例展示了如何操作

@Bean
public SimpleRabbitListenerContainerFactory rabbitListenerContainerFactory() {
    SimpleRabbitListenerContainerFactory factory = new SimpleRabbitListenerContainerFactory();
    ...
    factory.setMessageConverter(new Jackson2JsonMessageConverter());
    ...
    return factory;
}

这配置了一个 Jackson2 转换器,它需要存在头信息来指导转换。

还可以使用 ContentTypeDelegatingMessageConverter,它可以处理不同内容类型的转换。

从 2.3 版本开始,您可以通过在 messageConverter 属性中指定一个 bean 名称来覆盖工厂转换器。

@Bean
public Jackson2JsonMessageConverter jsonConverter() {
    return new Jackson2JsonMessageConverter();
}

@RabbitListener(..., messageConverter = "jsonConverter")
public void listen(String in) {
    ...
}

这避免了仅仅为了更改转换器而声明不同的容器工厂。

在大多数情况下,不需要自定义方法参数转换器,除非例如您想使用自定义的 ConversionService

在 1.6 版本之前,转换 JSON 的类型信息必须在消息头中提供,或者需要一个自定义的 ClassMapper。从 1.6 版本开始,如果没有类型信息头,可以从目标方法参数推断类型。

这种类型推断仅适用于方法级别的 @RabbitListener

有关更多信息,请参见 Jackson2JsonMessageConverter

如果您希望自定义方法参数转换器,可以按如下方式操作

@Configuration
@EnableRabbit
public class AppConfig implements RabbitListenerConfigurer {

    ...

    @Bean
    public DefaultMessageHandlerMethodFactory myHandlerMethodFactory() {
        DefaultMessageHandlerMethodFactory factory = new DefaultMessageHandlerMethodFactory();
        factory.setMessageConverter(new GenericMessageConverter(myConversionService()));
        return factory;
    }

    @Bean
    public DefaultConversionService myConversionService() {
        DefaultConversionService conv = new DefaultConversionService();
        conv.addConverter(mySpecialConverter());
        return conv;
    }

    @Override
    public void configureRabbitListeners(RabbitListenerEndpointRegistrar registrar) {
        registrar.setMessageHandlerMethodFactory(myHandlerMethodFactory());
    }

    ...

}
对于多方法监听器(参见 多方法监听器),方法选择基于消息转换之后的消息载荷。方法参数转换器仅在方法被选中后调用。