Spring AMQP 1.6完整参考指南

-第一部分

原文:http://docs.spring.io/spring-amqp/docs/1.6.0.RELEASE/reference/html/

1. 前言

 

Spring AMQP项目将其核心Spring概念应用于基于AMQP消息解决方案的开发中.我们提供了一个发送和接收消息的高级抽象模板.同时,我们也提供了消息驱动POJO的支持.这些包有助于AMQP资源的管理,从而提升依赖注入和声明式配置的使用. 在所有这些情况中,你会发现与Spring Framework对JMS支持是相似的. 对于其它项目相关的信息,可访问Spring AMQP项目主页

2. 介绍

本参考文档的第一部分是对Spring AMQP的高层次以及底层的概念的讲述,还有一些可使你尽快上手运行的代码片断.

2.1 快速浏览

2.1.1 介绍

5分钟开启Spring AMQP的旅行.

前提:

安装和运行RabbitMQ broker (http://www.rabbitmq.com/download.html). 然后获取spring-rabbit JAR 以及其依赖包 - 最简单的方式是在你的构建工具中声明依赖,如. 对于Maven来说:

org.springframework.amqp
spring-rabbit
1.6.0.RELEASE

对于gradle:

compile 'org.springframework.amqp:spring-rabbit:1.6.0.RELEASE'

 

兼容性

由于默认的Spring Framework 版本依赖是4.2.x, Spring AMQP 一般来说可兼容早期版本的Spring Framework.但基于注解的监听器和RabbitMessagingTemplate 需要Spring Framework 4.1+.

类似地,默认的amqp-client 版本是3.6.x,但framework 兼容3.4.0+.但依赖于新客户端版本的功能将无法使用.注意,这里指的是java client library; 一般来说,对于老版本的broker也可以工作.

非常非常快速

 

使用纯java来发送和接收消息:

ConnectionFactory connectionFactory = new CachingConnectionFactory(); 
AmqpAdmin admin = new RabbitAdmin(connectionFactory); 
admin.declareQueue(new Queue("myqueue")); 
AmqpTemplate template = new RabbitTemplate(connectionFactory); 
template.convertAndSend("myqueue", "foo"); 
String foo = (String) template.receiveAndConvert("myqueue");

注意,在原生的Java Rabbit client中也有一个ConnectionFactory. 在上面的代码中,我们使用的是Spring 抽象. 我们依赖的是broker中的默认交换器 (因为在发送操作中没有指定交换器),以及默认绑定(通过其名称作为路由键将所有队列绑定到默认交换器中).那些行为是由AMQP规范来定义的.

使用 XML配置

 

同上面的例子一样,只不过将外部资源配置到了XML:

ApplicationContext context =new GenericXmlApplicationContext("classpath:/rabbit-context.xml"); 
AmqpTemplate template = context.getBean(AmqpTemplate.class); 
template.convertAndSend("myqueue", "foo"); 
String foo = (String) template.receiveAndConvert("myqueue");
 
   
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:rabbit="http://www.springframework.org/schema/rabbit"
xsi:schemaLocation="http://www.springframework.org/schema/rabbit            
http://www.springframework.org/schema/rabbit/spring-rabbit.xsd            
http://www.springframework.org/schema/beans            
http://www.springframework.org/schema/beans/spring-beans.xsd">
 
   

 声明会默认自动查找类型为QueueExchange 和Binding的bean, 并宣称他们代表的broker的用户, 因此在简单的Java driver中没有必要明确的使用那个bean. 
有大量的选项来配置XML schema 中的组件属性- 你可以使用xml编辑的自动完成功能来探索它们并查看它们的相关文档.

使用 Java 配置


 

同样的例子可使用java来外部配置:

ApplicationContext context = new AnnotationConfigApplicationContext(RabbitConfiguration.class);
AmqpTemplate template = context.getBean(AmqpTemplate.class);
template.convertAndSend("myqueue", "foo");
String foo = (String) template.receiveAndConvert("myqueue");

........

@Configuration
public class RabbitConfiguration {

    @Bean
 public ConnectionFactory connectionFactory() {
        returnnew CachingConnectionFactory("localhost");
    }

    @Bean
 public AmqpAdmin amqpAdmin() {
returnnew RabbitAdmin(connectionFactory());
 }

    @Bean
 public RabbitTemplate rabbitTemplate() {
        returnnew RabbitTemplate(connectionFactory());
    }

    @Bean
 public Queue myQueue() {
       returnnew Queue("myqueue");
    }
}

2.2 新特性

2.2.1从1.5到1.6的变化

测试支持

 

 

提供了一个新的测试支持包. 参考Section 3.4, “Testing Support” 来了解更多信息.

Builder

 

 

现在Builders提供了更方便的API来配置Queue 和 Exchange 对象. 参考the section called “Builder API for Queues and Exchanges” 来了解更多信息.

命名空间变化

连接工厂

 

 

为连接工厂bean声明增加了thread-factory, 例如,以amqp-client包创建的线程命名.参考Section 3.1.2, “Connection and Resource Management” 来了解更多信息.

当使用CacheMode.CONNECTION时,现在你可以限制允许连接的总数了. 参考Section 3.1.2, “Connection and Resource Management” 来了解更多信息.

Queue 定义

 

 

现在为匿名队列提供了命名策略; 参考 the section called “AnonymousQueue” 来了解更多信息.

监听器容器变化

空闲消息监听器检测

 

 

当空闲时,现在可以配置监听器容器来发布ApplicationEvent 事件了. 参考the section called “Detecting Idle Asynchronous Consumers”来了解更多信息.
 

不匹配的队列检测

 

 

默认情况下,当监听器容器启动时,如果探测到了队列的mismatched属性或参数,容器会记录异常但会继续监听.现在容器有了一个mismatchedQueuesFatal 属性,当启动时存在问题的话,这可以阻止容器(以及上下文)启动.如果随后检测到问题,它也会停止容器,如从连接故障中恢复. 参考Section 3.1.15, “Message Listener Container Configuration” 来了解更多信息.

监听器容器日志

 

 

现在监听器容器提供了它的beanName 给内部 SimpleAsyncTaskExecutor 作为threadNamePrefix.对于日志分析非常有用.

默认错误处理器

 

 

默认错误处理器(ConditionalRejectingErrorHandler) 现在认为无可挽救的@RabbitListener 异常是致命的. 参考Section 3.1.13, “Exception Handling” 来了解更多信息.

AutoDeclare 与 RabbitAdmins

 

 

参考 Section 3.1.15, “Message Listener Container Configuration” (autoDeclare) 来查看在应用程序上下文使用RabbitAdmin s的语法变化.

AmqpTemplate: receive 与 timeout

 

 

AmqpTemplate 和它的RabbitTemplate实现引入了许多新的带有timeout的receive() 方法. 参考the section called “Polling Consumer”  来了解更多信息.

AsyncRabbitTemplate

 

 

引入了新的AsyncRabbitTemplate. 此模块提供了许多发送和接收的方法,其返回值为ListenableFuture,此后可通过它来同步或异步地获取结果. 参考 the section called “AsyncRabbitTemplate” 来了解更多信息

RabbitTemplate 变化

 

 

1.4.1 在broker支持时,引入了Direct reply-to 的能力;它比使用临时队列来回应更高效. 这个版本允许你覆盖这种默认行为,通常设置useTemporaryReplyQueues属性为true来使用临时队列. 参考 the section called “RabbitMQ Direct reply-to” 来了解更多信息.

RabbitTemplate 现在支持user-id-expression (userIdExpression 当使用Java配置时). 参考See Validated User-ID RabbitMQ documentationand the section called “Validated User Id” 来了解更多信息.

消息属性

CorrelationId

 

 

correlationId 消息属性现在可以是字符串了.参考the section called “Message Properties Converters” 来了解更多信息.

长字符串头

 

 

以前,DefaultMessagePropertiesConverter 会将头转换成(其头长度不超过长字符串限制(默认为1024) )成一个DataInputStream (实际上它只是引用LongString的 DataInputStream). 
在输出时,这个头不会进行转换(除了字符串, 例如. 在java.io.DataInputStream@1d057a39 上调用toString()方法).

在这个版本中, Long LongString 默认作为 LongString s ;你可以通过其getBytes[]toString(), 或 getStream()方法来访问它的内容. 更大的输入LongString 现在也会在输出上正确转换了.

参考 the section called “Message Properties Converters” 来了解更多信息.

Inbound Delivery Mode

 

 

deliveryMode 属性不再映射MessageProperties.deliveryMode;这是为了避免意外传播,如果同一个MessageProperties 对象用来发送出站消息. 
相反, 入站deliveryMode 头映射为MessageProperties.receivedDeliveryMode.

参考 the section called “Message Properties Converters” 来了解更多信息.

当使用注解endpoints时,头将在名为AmqpHeaders.RECEIVED_DELIVERY_MODE的头中提供.

参考the section called “Annotated Endpoint Method Signature” 来了解更多信息.

Inbound User ID

 

 

user_id 属性不再映射MessageProperties.userId; 这是为了避免意外传播,如果同一个MessageProperties 对象用来发送出站消息.相反, 入站userId 头会映射到MessageProperties.receivedUserId.

参考 the section called “Message Properties Converters” 来了解更多信息.

当使用注解endpoints时,头将在名为AmqpHeaders.RECEIVED_USER_ID的头中提供.

参考 the section called “Annotated Endpoint Method Signature”  来了解更多信息.

RabbitAdmin 变化

Declaration Failures

 

 

之间,ignoreDeclarationFailures 标志只在channel上发生IOException 才会生效(如未匹配参数). 现在它对于任何异常都可以生效(如:TimeoutException). 
此外,无论何时声明失败,都会发布DeclarationExceptionEvent.RabbitAdmin 最后声明事件将作为属性lastDeclarationExceptionEvent也是可用的. 

参考 Section 3.1.10, “Configuring the broker” 来了解更多信息.

@RabbitListener Changes

Multiple Containers per Bean

 

 

当使用Java 8+,可以添加多个@RabbitListener 注解到 @Bean 类或它们的方法上.当使用Java 7-,你可以使用 @RabbitListeners 容器注解来提供同样的功能. 
参考the section called “@Repeatable @RabbitListener” 来了解更多信息.

@SendTo SpEL Expressions

 

 

@SendTo for routing replies with no replyTo property can now be SpEL expressions evaluated against the request/reply. 

参考 the section called “Reply Management”来了解更多信息.

@QueueBinding Improvements

 

 

现在你可以在@QueueBinding 注解中为queues, exchanges 和bindings 指定参数. 

Header交换器可通过@QueueBinding来支持

参考the section called “Annotation-driven Listener Endpoints” 来了解更多信息.

延迟消息交换器(Delayed Message Exchange)

 

 

Spring AMQP 现在有了第一个支持RabbitMQ Delayed Message Exchange 插件. 参考Section 3.1.11, “Delayed Message Exchange” 来了解更多信息.

交换器内部标志(Exchange internal flag)

 

 

任何Exchange 定义现在都可标记为internal ,当声明交换器时,RabbitAdmin 会将值传递给broker. 

参考 Section 3.1.10, “Configuring the broker”来了解更多信息.

CachingConnectionFactory 变化

CachingConnectionFactory Cache Statistics

 

 

 CachingConnectionFactory 现在通过运行时和JMX提供了cache属性.参考the section called “Runtime Cache Properties”  来了解更多信息.

访问底层RabbitMQ连接工厂

 

 

添加了一个新的getter方法来获取底层factory.这是很有用的,例如,可以添加自定义属性. 参考 Section 3.1.3, “Adding Custom Client Connection Properties” 来了解更多信息.

Channel Cache

 

 

默认的channel 缓存大小已从1增加到了25. 参考Section 3.1.2, “Connection and Resource Management”来了解更多信息.

此外, SimpleMessageListenerContainer 不再设置缓存大小至少要与concurrentConsumers的数目一样大 - 这是多余的,因为容器消费者channels是不会缓存的.

RabbitConnectionFactoryBean

 

 

factory bean现在暴露了一个属性来将client连接属性添加到由工厂产生的连接中.

Java 反序列化(Deserialization)

 

 

当使用Java反序列化时,现在允许配置类的白名单. 如果你从不可信来源接收消息,考虑创建白名单是很重要的. 参考 the section called “Java Deserialization” 来了解更多信息.

JSON MessageConverter

 

 

改善了JSON消息转换器,现在允许消费消息在消息头中可以没有类型(type)信息. 参考the section called “Message Conversion for Annotated Methods” 和 the section called “Jackson2JsonMessageConverter”来了解更多信息.

Logging Appenders

Log4j2

 

 

加入了log4j2 appender,此appenders现在可配置addresses 属性来连接broker集群.

Client 连接属性

 

 

现在你可以向RabbitMQ连接中加入自定义连接属性了.

参考Section 3.2, “Logging Subsystem AMQP Appenders” 来了解更多信息.

2.2.2 早期版本

 

 

参考Section A.2, “Previous Releases” 来了解早期版本的变化.

 

-第二部分

3. 参考

这部分参考文档详细描述了组成Sring AMQP的各种组件. main chapter 涵盖了开发AMQP应用程序的核心类. 这部分也包含了有关示例程序的章节.

3.1 使用 Spring AMQP

在本章中,我们将探索接口和类,它们是使用Spring AMQP来开发应用程序的必要组件 .

3.1.1 AMQP 抽象

介绍

 

Spring AMQP 由少数几个模块组成, 每个都以JAR的形式来表现.这些模块是: spring-amqp和spring-rabbit. spring-amqp模块包含org.springframework.amqp.core 包. 
在那个包中,你会找到表示AMQP核心模块的类. 我们的目的是提供通用的抽象,不依赖于任何特定中间件的实现或AMQP客户端库。
最终用户代码将更具有移植性,以便跨供应商实现,因为它们可以对抽象层开发。这些抽象是使用broker的特定模块实现的,如spring-rabbit。
目前只有一个RabbitMQ实现;而对于抽象的验证,除了RabbitMQ外,也已经在.Net平台上使用Apache Qpid得到了验证。
由于AMQP原则上工作于协议层次,RabbitMQ客户端可以在任何支持相同的协议版本的broker中使用,但目前我们没有测试其它任何broker。

这里的概述假设你已经熟悉了AMQP规范.如果你还没有,那么你可以查看第5章,其它资源中列举的资源.
 

Message

 

 

AMQP 0-8 和0-9-1 规范没有定义一个消息类或接口.相反,当执行basicPublish()这样的操作的时候, 内容是以字节数组参数进行传递的,其它额外属性也是以单独参数进行传递的. Spring AMQP定义了一个 Message 类来作为AMQP领域模型表示的一部分.Message类的目的是在单个实例中简化封装消息体(body)和属性(header),这样API就可以变得更简单. Message类的定义相当简单.
 

public class Message {

    private final MessageProperties messageProperties;

    private final byte[] body;

    public Message(byte[] body, MessageProperties messageProperties) {
        this.body = body;
        this.messageProperties = messageProperties;
    }

    public byte[] getBody() {
        returnthis.body;
    }

    public MessageProperties getMessageProperties() {
        returnthis.messageProperties;
    }
}

MessageProperties 接口定义了多个共同属性,如messageIdtimestampcontentType 等等. 那些属性可以通过调用setHeader(String key, Object value) 方法使用用户定义头(user-defined headers)来扩展.

 

Exchange

 

Exchange 接口代表的是AMQP Exchange,它是生产者发送消息的地方.broker虚拟主机中的交换器名称都是唯一的,同时还有少量的其它属性:

public interface Exchange {

    String getName();

    String getExchangeType();

    boolean isDurable();

    boolean isAutoDelete();

    Map getArguments();

}

正如你所看到的, Exchange还有一个type (它是在ExchangeTypes中定义的常量). 基本类型是: DirectTopicFanout,和Headers.
在核心包中,你可以找到每种类型的Exchange 接口实现.这些交换器类型会在处理队列绑定时,行为有所不同.
例如,Direct交换器允许队列以固定路由键进行绑定(通常是队列的名称).
Topic交换器支持使用路由正则表达式(*通配符明确匹配一个,而#通配符可匹配0个或多个). 

Fanout交换器会把消息发布到所有绑定到它上面的队列而不考虑任何路由键.

关于交换器类型的更多信息,查看Chapter 5, Other Resources.

AMQP规范还要求任何broker必须提供一个默认的无名字的(空字符串)Direct交换器.所有声明的队列都可以用它们的名字作为路由键绑定到默认交换器中. 在Section 3.1.4, “AmqpTemplate”你会了解到更多关于在Sring AMQP中使用默认交换器的使用情况.
 

Queue

 

Queue 代表的是消费者接收消息的组件. 像各种各样的 Exchange 类,我们的实现目标是作为核心AMQP类型的抽象表示.

public class Queue  {

    private final String name;

    private volatile boolean durable;

    private volatile boolean exclusive;

    private volatile boolean autoDelete;

    private volatile Map arguments;

    /**
     * The queue is durable, non-exclusive and non auto-delete.
     *
     * @param name the name of the queue.
     */
 public Queue(String name) {
        this(name, true, false, false);
    }

    // Getters and Setters omitted for brevity

}

注意,构造器需要接受队列名称作为参数.根据实现, admin template可能会提供生成独特队列名称的方法.这些队列作为回复地址或用于临时情景是非常有用的.
基于这种原因,自动生成队列的exclusive和 autoDelete 属性都应该设置为true.

参考 Section 3.1.10, “Configuring the broker” 来了解关于使用命名空间来声明队列,包括队列参数的详细情况.

Binding

 

生产者发送消息到Exchange,而消费者将从Queue中获取消息,连接Queues与Exchanges之间的绑定对于通过消息来连接生产者和消费者是非常关键的.
在Spring AMQP中,我们定义了一个 Binding 类来表示这些连接. 让我们重新回顾一下绑定队列和交换器的操作.

你可以使用固定的路由键来绑定 Queue 到 DirectExchange上.

new Binding(someQueue, someDirectExchange, "foo.bar")

你可以使用路由正则表达式来绑定Queue到TopicExchange上.

new Binding(someQueue, someTopicExchange, "foo.*")

你可以不使用路由键来绑定Queue到FanoutExchange上.

new Binding(someQueue, someFanoutExchange)

我们还提供了BindingBuilder来方便操作.

Binding b = BindingBuilder.bind(someQueue).to(someTopicExchange).with("foo.*");

上面展示的BindingBuilder 类很清晰,但如果为bind()方法使用静态导入,这种形式将工作得更好.

本身来说,Binding类的实例只能一个connection中持有数据.换句话说,它不是一个活力(active)组件.
但正如在后面Section 3.1.10, “Configuring the broker”看到的, Binding实例可由AmqpAdmin 类来触发broker上的绑定操作.
同样,在同一个章节中,你还会看到Binding实例可在@Configuration类中使用Spring @Bean风格来定义. 
还有方便的基类来简化生成AMQP相关bean定义和识别队列,交换器,绑定的方法,这样当AMQP broker运行程序启动时,就可以得到声明.

AmqpTemplate 也在核心包中定义.作为涉及AMQP消息的主要组件, 会在它自己的章节中进行详细介绍(参考Section 3.1.4, “AmqpTemplate”).

3.1.2 连接和资源管理

介绍

 

虽然我们在前面章节中描述的AMQP模型是通用的,适用于所有实现,但当我们说到资源管理时,其细节是针对特定broker实现的.因此,在这个章节中,我们只关注我们的"spring-rabbit"模块,因为到现在为止,RabbitMQ是唯一支持的实现.

RabbitMQ broker中用于管理连接的中心组件是ConnectionFactory 接口. ConnectionFactory实现的责任是提供一个org.springframework.amqp.rabbit.connection.Connection 的实例,它包装了com.rabbitmq.client.Connection
我们提供的唯一具体实现提CachingConnectionFactory,默认情况下,会建立应用程序可共享的单个连接代理.连接共享是可行的,因为在AMQP处理消息的工作单位实际是 "channel" (在某些方面,这类似于JMS中Connection 和 Sessionin的关系).
你可以想象,连接实例提供了一个createChannel方法。CachingConnectionFactory 实现支持这些channels的缓存,它会基于它们是否是事务的来单独维护其缓存. 
当创建CachingConnectionFactory的实例时hostname 可通过构造器来提供,username 和password 属性也可以提供.如果你想配置channel缓存的大小(默认是25),你可以调用setChannelCacheSize()方法.

1.3版本开始,CachingConnectionFactory 也可以同channel一样,配置缓存连接.在这种情况下每次调用createConnection() 都会创建一个新连接(或者从缓存中获取空闲的连接).
关闭连接会将其返回到缓存中(如果还没有达到缓存大小的话).在这些连接上创建的Channels同样也会被缓存. 单独连接的使用在某些环境中是有用的,如从HA 集群中消费, 连接负载均衡器,连接不同的集群成员.设置cacheMode 为 CacheMode.CONNECTION.

这不会限制连接的数目,它用于指定允许空闲打开连接的数目.

从1.5.5版本开始,提供了一个新属性connectionLimit.当设置了此属性时,它会限制连接的总数目,当达到限制值时,将channelCheckoutTimeLimit 来等待空闲连接.如果时间超时了,将抛出AmqpTimeoutException.

重要

当缓存模式是CONNECTION时, 队列的自动声明等等 (参考 the section called “Automatic Declaration of Exchanges, Queues and Bindings”) 将不再支持.

 

此外,在写作的时候,rabbitmq-client 包默认为每个连接(5个线程)创建了一个固定的线程池. 当需要使用大量连接时,你应该考虑在CachingConnectionFactory定制一个executor. 然后,同一个executor会用于所有连接,其线程也是共享的.  
executor的线程池是没有界限的或按预期使用率来设置(通常, 一个连接至少应该有一个线程).如果在每个连接上创建了多个channels,那么池的大小会影响并发性,因此一个可变的线程池executor应该是最合适的.

 

理解缓存大小不是限制是很重要的, 它仅仅是可以缓存的通道数量.当说缓存大小是10时,在实际使用中,其实可以是任何数目的通道. 如果超过10个通道被使用,他们都返回到高速缓存,10个将在高速缓存中,其余的将物理关闭。

1.6版本开始,默认通道缓存大小从1增加到了25. 在高容量,多线程环境中,较小的缓存意味着通道的创建和关闭将以很高的速率运行.加大默认缓存大小可避免这种开销.
你可以监控通道的使用情况(通过RabbitMQ Admin UI) ,如果看到有很多通道在创建和关闭,你可以增大缓存大小.缓存只会增长按需(以适应应用程序的并发性要求),所以这个更改不会影响现有的低容量应用程序。

1.4.2版本开始,CachingConnectionFactory 有一个channelCheckoutTimeout属性. 当此属性的值大于0时,channelCacheSize 会变成连接上创建通道数目的限制.
如果达到了限制,调用线程将会阻塞,直到某个通道可用或者超时, 在后者的情况中,将抛出AmqpTimeoutException 异常.

在框架(如.RabbitTemplate)中使用的通道将会可靠地返回到缓存中.如果在框架外创建了通道 (如.直接访问connection(s)并调用createChannel()),你必须可靠地返回它们(通过关闭),也许需要在 finally 块中以防止耗尽通道.

CachingConnectionFactory connectionFactory = new CachingConnectionFactory("somehost");
connectionFactory.setUsername("guest");
connectionFactory.setPassword("guest");

Connection connection = connectionFactory.createConnection();

当使用XML时,配置看起来像下面这样:

 
   
class="org.springframework.amqp.rabbit.connection.CachingConnectionFactory">
 
   

这里也有一个 SingleConnectionFactory 实现,它只能用于框架的单元测试代码中.它比CachingConnectionFactory 简单,因为它不会缓存通道,由于其缺乏性能和韧性,它不打算用于简单的测试以外的实际使用. 
如果基于某些原因,你需要自己来实现ConnectionFactory ,AbstractConnectionFactory 基类提供了一个非常好的起点.

ConnectionFactory 可使用rabbit命名空间来快速方便的建立:

在多数情况下,这是很好的,因为框架会为你选择最好的默认值.创建的实例会是CachingConnectionFactory.要记住,默认的缓存大小是25.如果你想缓存更多的通道,你可以设置channelCacheSize 属性值.在XML中,它看起来像下面这样:

 
   
class="org.springframework.amqp.rabbit.connection.CachingConnectionFactory">
 
   

在命名空间中,你也可以添加channel-cache-size 属性:

默认的缓存模式是CHANNEL, 但你可以使用缓存连接来替换;在这种情况下,我们会使用connection-cache-size:

Host 和 port 属性也可以在命名空间中提供:

此外,如果运行集群环境中,使用addresses属性.

下面是一个自定义的线程工厂,其前辍线程名称为rabbitmq-.

 
    
   

配置底层客户端连接工厂

 

CachingConnectionFactory 使用的是 Rabbit client ConnectionFactory的实例; 当在CachingConnectionFactory设置等价属性时,许多属性(host, port, userName, password, requestedHeartBeat, connectionTimeout) 来传递. 
要设置其它属性(例如clientProperties),可定义一个rabbit factory 的实例,并使用CachingConnectionFactory的适当构造器来提供引用. 
当使用上面提到的命名空间时,要在connection-factory属性中提供一个工厂的引用来配置. 为方便起见,提供了一个工厂,以协助在一个Spring应用程序上下文中配置连接工厂,在下一节讨论。

RabbitConnectionFactoryBean 和配置SSL

 

1.4版本开始, 提供了一个便利的RabbitConnectionFactoryBean 类通过依赖注入来配置底层客户端连接工厂的SSL属性.其它设置简单地委派给底层工厂.以前你必须以编程方式配置SSL选项。

 
   

参考 RabbitMQ Documentation 来了解关于配置SSL的更多信息. 省略的keyStore 和 trustStore 配置将在无证书验证的情况下,通过SSL来连接. Key和trust store 配置可以按如下提供:

sslPropertiesLocation 属性是一个Spring Resource ,它指向一个包含下面key的属性文件:

keyStore=file:/secret/keycert.p12
trustStore=file:/secret/trustStore
keyStore.passPhrase=secret
trustStore.passPhrase=secret

keyStore 的 truststore 是指向store的 Spring Resources .通常情况下,这个属性文件在操作系统之下安全的,应用程序只能读取访问.

从Spring AMQP 1.5版本开始,这些属性可直接在工厂bean上设置.如果同时提供了discrete和 sslPropertiesLocation 属性, 后者属性值会覆盖discrete值.

路由连接工厂

 

1.3版本开始,引入了AbstractRoutingConnectionFactory.这提供了一种机制来配置多个ConnectionFactories的映射,并通过在运行时使用lookupKey来决定目标ConnectionFactory

通常,实现会检查线程绑定上下文. 为了方便, Spring AMQP提供了SimpleRoutingConnectionFactory, 它会从SimpleResourceHolder中获取当前线程绑定的lookupKey:

 
 
   
 
   
 
   
public class MyService {

	@Autowired
  private RabbitTemplate rabbitTemplate;

	public  void service(String vHost, String payload) {
		SimpleResourceHolder.bind(rabbitTemplate.getConnectionFactory(), vHost);
		rabbitTemplate.convertAndSend(payload);
		SimpleResourceHolder.unbind(rabbitTemplate.getConnectionFactory());
	}

}

在使用资源后,对其进行解绑是很重要的.更多信息参考AbstractRoutingConnectionFactory的JavaDocs.

1.4版本开始RabbitTemplate 支持SpEL sendConnectionFactorySelectorExpression 和receiveConnectionFactorySelectorExpression 属性, 
它会在每个AMQP 协议交互操作(sendsendAndReceivereceiveor receiveAndReply)进行评估, 为提供的AbstractRoutingConnectionFactory类解析lookupKey值
Bean 引用,如"@vHostResolver.getVHost(#root)" 可用于表达式中.对于send 操作,  要发送的消息是根评估对象;对于receive操作queueName 是根评估对象.

路由算法为:如果selector 表达式为null,或等价于null,或提供的ConnectionFactory 不是AbstractRoutingConnectionFactory的实例,根据提供的ConnectionFactory 实现,
所有的工作都按之前的进行.同样的结果也会发生:如果评估结果不为null,但对于lookupKey 无目标ConnectionFactory,且 the AbstractRoutingConnectionFactory 使用lenientFallback = true进行了配置
当然,在AbstractRoutingConnectionFactory 的情况下,它会基于determineCurrentLookupKey()的路由实现来进行回退. 但,如果lenientFallback = false, 将会抛出 IllegalStateException 异常.

Namespace 在组件中也支持send-connection-factory-selector-expression 和receive-connection-factory-selector-expression属性.

也是从1.4版本开始, 你可以在SimpleMessageListenerContainer配置路由连接工厂. 在那种情况下,队列名称的列表将作为lookup key.例如,如果你在容器中配置setQueueNames("foo", "bar"),lookup key将是"[foo,bar]" (无空格).

 

-第三部分

Queue Affinity 和 LocalizedQueueConnectionFactory

当在集群中使用HA队列时,为了获取最佳性能,可以希望连接到主队列所在的物理broker. 虽然CachingConnectionFactory 可以配置为使用多个broker 地址; 这会失败的,client会尝试按顺序来连接. LocalizedQueueConnectionFactory 使用管理插件提供的 REST API来确定包含master队列的节点.然后,它会创建(或从缓存中获取)一个只连接那个节点的CachingConnectionFactory .如果连接失败了,将会确定一个新的消费者可连接的master节点. LocalizedQueueConnectionFactory 使用默认的连接工厂进行配置,在队列物理位置不能确定的情况下,它会按照正常情况来连接集群.

LocalizedQueueConnectionFactory 是一个RoutingConnectionFactory , SimpleMessageListenerContainer 会使用队列名称作为其lookup key ,这些已经在上面的 the section called “Routing Connection Factory” 讨论过了.

 

基于这个原因(使用队列名称来作查找键),LocalizedQueueConnectionFactory 只在容器配置为监听某个单一队列时才可使用.

RabbitMQ 管理插件应该在每个节点上开启.

警告

这种连接工厂用于长连接,如用在SimpleMessageListenerContainer的连接.它的目的不是用于短连接, 如在 RabbitTemplate中使用,这是因为在连接前,它要调用REST API. 此外,对于发布操作来说,队列是未知的,不管如何, 消息会发布到所有集群成员中,因此查找节点的逻辑几乎没有什么意义。

这里有一个样例配置,使用了Spring Boot的RabbitProperties来配置工厂:

@Autowired
private RabbitProperties props;

private final String[] adminUris = { "http://host1:15672", "http://host2:15672" };

private final String[] nodes = { "rabbit@host1", "rabbit@host2" };

@Bean
public ConnectionFactory defaultConnectionFactory() {
    CachingConnectionFactory cf = new CachingConnectionFactory();
    cf.setAddresses(this.props.getAddresses());
    cf.setUsername(this.props.getUsername());
    cf.setPassword(this.props.getPassword());
    cf.setVirtualHost(this.props.getVirtualHost());
    return cf;
}

@Bean
public ConnectionFactory queueAffinityCF(
        @Qualifier("defaultConnectionFactory") ConnectionFactory defaultCF) {
       return new LocalizedQueueConnectionFactory(defaultCF,
            StringUtils.commaDelimitedListToStringArray(this.props.getAddresses()),
            this.adminUris, this.nodes,
            this.props.getVirtualHost(), this.props.getUsername(), this.props.getPassword(),
            false, null);
}

注意,三个参数是 addressesadminUris 和 nodes的数组. 当一个容器试图连接一个队列时,它们是有位置性的,它决定了哪个节点上的队列是mastered,并以同样数组位置来连接其地址.

发布者确认和返回

确认和返回消息可通过分别设置CachingConnectionFactory的 publisherConfirms 和publisherReturns 属性为ture来完成.

当设置了这些选项时,由工厂创建的通道将包装在PublisherCallbackChannel中,这用来方便回调. 当获取到这样的通道时,client可在channel上注册一个 PublisherCallbackChannel.ListenerPublisherCallbackChannel 实现包含一些逻辑来路由确认/返回给适当的监听器. 这些特性将在下面的章节中进一步解释.

对于一些更多的背景信息, 可以参考下面的博客:Introducing Publisher Confirms.

记录通道关闭事件

1.5版本中引入了允许用户控制日志级别的机制.

CachingConnectionFactory 使用默认的策略来记录通道关闭事件:

  • 不记录通道正常关闭事件 (200 OK).
  • 如果通道是因为失败的被动的队列声明关闭的,将记录为debug级别.
  • 如果通道关闭是因为basic.consume因专用消费者条件而拒绝引起的,将被记录为INFO级别.
  • 所有其它的事件将记录为ERROR级别.

要修改此行为,需要在CachingConnectionFactory的closeExceptionLogger属性中注入一个自定义的ConditionalExceptionLogger.

也可参考the section called “Consumer Failure Events”.

运行时缓存属性

从1.6版本开始CachingConnectionFactory 通过getCacheProperties()方法提供了缓存统计. 这些统计数据可用来在生产环境中优化缓存.例如, 最高水位标记可用来确定是否需要加大缓存.如果它等于缓存大小,你也许应该考虑进一步加大.

Table 3.1. CacheMode.CHANNEL的缓存属性

Property Meaning
channelCacheSize

当前配置的允许空闲的最大通道数量.

localPort

连接的本地端口(如果可用的话). 在可以在RabbitMQ 管理界面中关联 connections/channels.

idleChannelsTx

当前空闲(缓存的)的事务通道的数目.

idleChannelsNotTx

当前空闲(缓存的)的非事务通道的数目.

idleChannelsTxHighWater

同时空闲(缓存的)的事务通道的最大数目

idleChannelsNotTxHighWater

同时空闲(缓存的)的非事务通道的最大数目.

Table 3.2. CacheMode.CONNECTION的缓存属性

Property Meaning
openConnections

表示连接到brokers上连接对象的数目.

channelCacheSize

当前允许空闲的最大通道数目

connectionCacheSize

当前允许空闲的最大连接数目.

idleConnections

当前空闲的连接数目.

idleConnectionsHighWater

目前已经空闲的最大连接数目.

idleChannelsTx:

在当前连接上目前空闲的事务通道的数目. 属性名的localPort部分可用来在RabbitMQ 管理界面中关联connections/channels.

idleChannelsNotTx:

在当前连接上目前空闲和非事务通道的数目.属性名的localPort部分可用来在RabbitMQ管理界面中关联connections/channels 

idleChannelsTxHighWater:

已同时空闲的事务通道的最大数目. 属性名的 localPort部分可用来在RabbitMQ管理界面中关联connections/channels.

idleChannelsNotTxHighWater:

忆同时空闲的非事务通道的最大数目.属性名的localPort部分可用来RabbitMQ管理界面中关联connections/channels.

 

cacheMode 属性 (包含CHANNEL 或 CONNECTION ).

Figure 3.1. JVisualVM Example

Spring AMQP 1.6完整参考指南_第1张图片

3.1.3 添加自定义Client 连接属性

CachingConnectionFactory 现在允许你访问底层连接工厂,例如, 设置自定义client 属性:

connectionFactory.getRabbitConnectionFactory().getClientProperties().put("foo", "bar");

当在RabbitMQ管理界面中查看连接时,将会看到这些属性.

 

3.1.4 AmqpTemplate

介绍

像其它Spring Framework提供的高级抽象一样, Spring AMQP 提供了扮演核心角色的模板. 定义了主要操作的接口称为AmqpTemplate. 这些操作包含了发送和接收消息的一般行为.换句话说,它们不是针对某个特定实现的,从其名称"AMQP"就可看出.另一方面,接口的实现会尽量作为AMQP协议的实现.不像JMS,它只是接口级别的API实现, AMQP是一个线路级协议.协议的实现可提供它们自己的client libraries, 因此模板接口的实现都依赖特定的client library.目前,只有一个实现:RabbitTemplate. 在下面的例子中,你会经常看到"AmqpTemplate",但当你查看配置例子或者任何实例化或调用setter方法的代码时,你都会看到实现类型(如."RabbitTemplate").

正如上面所提到的, AmqpTemplate 接口定义了所有发送和接收消息的基本操作. 我们将分别在以下两个部分探索消息发送和接收。

也可参考the section called “AsyncRabbitTemplate”.

 

添加重试功能

从1.3版本开始, 你可为RabbitTemplate 配置使用 RetryTemplate 来帮助处理broker连接的问题. 参考spring-retry 项目来了解全部信息;下面就是一个例子,它使用指数回退策略(exponential back off policy)和默认的 SimpleRetryPolicy (向调用者抛出异常前,会做三次尝试).

使用XML命名空间:

 
   
 
   
 
   

使用 @Configuration:

@Bean
public AmqpTemplate rabbitTemplate();
		RabbitTemplate template = new RabbitTemplate(connectionFactory());
		RetryTemplate retryTemplate = new RetryTemplate();
		ExponentialBackOffPolicy backOffPolicy = new ExponentialBackOffPolicy();
		backOffPolicy.setInitialInterval(500);
		backOffPolicy.setMultiplier(10.0);
		backOffPolicy.setMaxInterval(10000);
		retryTemplate.setBackOffPolicy(backOffPolicy);
		template.setRetryTemplate(retryTemplate);
		return template;
}

从1.4版本开始,除了retryTemplate 属性外,RabbitTemplate 上也支持recoveryCallback 选项. 它可用作RetryTemplate.execute(RetryCallback retryCallback, RecoveryCallbackrecoveryCallback)第二个参数.

RecoveryCallback 会有一些限制,因为在retry context只包含lastThrowable 字段.在更复杂的情况下,你应该使用外部RetryTemplate,这样你就可以通过上下文属性传递更多信息给RecoveryCallback

retryTemplate.execute(
    new RetryCallback() {

        @Override
  public Object doWithRetry(RetryContext context) throws Exception {
            context.setAttribute("message", message);
            return rabbitTemplate.convertAndSend(exchange, routingKey, message);
        }
    }, new RecoveryCallback() {

        @Overridepublic Object recover(RetryContext context) throws Exception {
            Object message = context.getAttribute("message");
            Throwable t = context.getLastThrowable();
            // Do something with message 
   
   return null;
        }
    });
}

在这种情况下,你不需要在RabbitTemplate中注入RetryTemplate.

发布者确认和返回

AmqpTemplate的RabbitTemplate 实现支持发布者确认和返回.

对于返回消息,模板的 mandatory 属性必须设置为true, 或者对于特定消息,其 mandatory-expression 必须评估为true .
此功能需要将CachingConnectionFactory 的publisherReturns 属性设置为true (参考 the section called “Publisher Confirms and Returns”).
返回是通过注册在RabbitTemplate.ReturnCallback(通过调用setReturnCallback(ReturnCallback callback))来返回给客户端的. 回调必须实现下面的方法:

void returnedMessage(Message message, int replyCode, String replyText, String exchange, String routingKey);

每个RabbitTemplate只支持一个ReturnCallback .也可参考the section called “Reply Timeout”.

对于发布者确认(又名发布者应答), 模板需要将 CachingConnectionFactory 中的publisherConfirms 属性设置为true.
确认是通过注册在RabbitTemplate.ConfirmCallback(通过调用setConfirmCallback(ConfirmCallback callback)) 发送给client的. 回调必须实现下面的方法:

void confirm(CorrelationData correlationData, boolean ack, String cause);

CorrelationData 对象是在发送原始消息的时候,由client提供的. ack 为true 表示确认,为false时,表示不确认(nack). 对于nack , cause可能会包含nack的原因(如果生成nack时,它可用的话). 
一个例子是当发送消息到一个不存在的交换器时.在那种情况下,broker会关闭通道; 关闭的原因会包含在cause中cause 是1.4版本中加入的.

RabbitTemplate中只支持一个ConfirmCallback.

当rabbit模板完成发送操作时,会关闭通道; 这可以排除当连接工厂缓存满时(缓存中还有空间,通道没有物理关闭,返回/确认正常处理)确认和返回的接待问题.
当缓存满了的时候, 框架会延迟5秒来关闭,以为接收确认/返回消息留有时间.当使用确认时,通道会在收到最后一个确认时关闭.
当使用返回时,通道会保持5秒的打开状态.一般建议将连接工厂的channelCacheSize 设为足够大,这样发布消息的通道就会返回到缓存中,而不是被关闭.
你可以使用RabbitMQ管理插件来监控通道的使用情况;如果你看到通道打开/关闭的非常迅速,那么你必须考虑加大缓存,从而减少服务器的开销.

Messaging 集成

从1.4版本开始, 构建于RabbitTemplate上的RabbitMessagingTemplate提供了与Spring Framework消息抽象的集成(如.org.springframework.messaging.Message)
This allows you to create the message to send in generic manner.

验证 User Id

从1.6版本开始,模板支持user-id-expression (当使用Java配置时,为userIdExpression). 如果发送消息,user id属性的值将在评估表达式后进行设置.评价的根对象是要发送的消息。

例子:

第一个示例是一个文本表达式;第二个例子将获取上下文中连接工厂bean的username 属性.

3.1.5 发送消息

介绍

当发送消息时,可使用下面的任何一种方法:

void send(Message message) throws AmqpException;

void send(String routingKey, Message message) throws AmqpException;

void send(String exchange, String routingKey, Message message) throws AmqpException;

我们将使用上面列出的最后一个方法来讨论,因为它实际是最清晰的.它允许在运行时提供一个AMQP Exchange 名称和路由键(routing key).最后一个参数是负责初建创建Message实例的回调.使用此方法来发送消息的示例如下:

amqpTemplate.send("marketData.topic", "quotes.nasdaq.FOO",
    new Message("12.34".getBytes(), someProperties));

如果你打算使用模板实例来多次(或多次)向同一个交换器发送消息时,"exchange" 可设置在模板自已身上.在这种情况中,可以使用上面列出的第二个方法. 下面的例子在功能上等价于前面那个:

amqpTemplate.setExchange("marketData.topic");
amqpTemplate.send("quotes.nasdaq.FOO", new Message("12.34".getBytes(), someProperties));

如果在模块上设置"exchange"和"routingKey"属性,那么方法就只接受Message 参数:

amqpTemplate.setExchange("marketData.topic");
amqpTemplate.setRoutingKey("quotes.nasdaq.FOO");
amqpTemplate.send(new Message("12.34".getBytes(), someProperties));

关于交换器和路由键更好的想法是明确的参数将总是会覆盖模板默认值.事实上, 即使你不在模板上明确设置这些属性, 总是有默认值的地方. 在两种情况中,默认值是空字符串,这是合情合理的. 
就路由键而言,它并不总是首先需要的 (如. Fanout 交换器). 此外,绑定的交换器上的队列可能会使用空字符串. 这些在模板的路由键中都是合法的. 
就交换器名称而言,空字符串也是常常使用的,因为AMQP规范定义了无名称的"默认交换器".
由于所有队列可使用它们的队列名称作为路由键自动绑定到默认交换器上(它是Direct交换器e) ,上面的第二个方法可通过默认的交换器将简单的点对点消息传递到任何队列. 
只需要简单的将队列名称作为路由键-在运行时提供方法参数:

RabbitTemplate template = new RabbitTemplate(); // 使用默认的无名交换器
template.send("queue.helloWorld", new Message("Hello World".getBytes(), someProperties));

或者,如果你喜欢创建一个模板用于主要或专门向一个队列发送消息, 以下是完全合理的:

RabbitTemplate template = new RabbitTemplate(); // 使用默认无名交换器
template.setRoutingKey("queue.helloWorld"); // 但我们总是向此队列发送消息
template.send(new Message("Hello World".getBytes(), someProperties));

Message Builder API

1.3版本开始,通过 MessageBuilder 和 MessagePropertiesBuilder提供了消息构建API; 它们提供了更加方便地创建消息和消息属性的方法:

Message message = MessageBuilder.withBody("foo".getBytes())
	.setContentType(MessageProperties.CONTENT_TYPE_TEXT_PLAIN)
	.setMessageId("123")
	.setHeader("bar", "baz")
	.build();

MessageProperties props = MessagePropertiesBuilder.newInstance()
	.setContentType(MessageProperties.CONTENT_TYPE_TEXT_PLAIN)
	.setMessageId("123")
	.setHeader("bar", "baz")
	.build();
Message message = MessageBuilder.withBody("foo".getBytes())
	.andProperties(props)
	.build();

每个MessageProperies上定义的属性都可以被设置. 其它方法包括setHeader(String key, String value),removeHeader(String key)removeHeaders(), 和copyProperties(MessageProperties properties)
每个属性方法都有一个set*IfAbsent() 变种. 在默认的初始值存在的情况下, 方法名为set*IfAbsentOrDefault().

提供了五个静态方法来创建初始message builder:

public static MessageBuilder withBody(byte[] body)
public static MessageBuilder withClonedBody(byte[] body)
public static MessageBuilder withBody(byte[] body, int from, int to)
public static MessageBuilder fromMessage(Message message)
public static MessageBuilder fromClonedMessage(Message message)

builder创建的消息body是参数的直接引用.

builder创建的消息body是包含拷贝原字节数组的新数组.

build创建的消息body是包含原字节数组范围的新数组.查看Arrays.copyOfRange() 来了解更多信息.

builder创建的消息body是原body参数的直接引用. 参数的属性将拷贝到新MessageProperties对象中.

builer创建的消息body包含参数body的新数组.参数的属性将拷贝到新的MessageProperties 对象中.

public static MessagePropertiesBuilder newInstance()
public static MessagePropertiesBuilder fromProperties(MessageProperties properties)
public static MessagePropertiesBuilder fromClonedProperties(MessageProperties properties)

新消息属性将使用默认值进行初始化

builder会使用提供的properties对象进行初始化,build() 方法也会返回参数properties对象.

参数的属性会拷贝到新的MessageProperties对象中.

在AmqpTemplate的RabbitTemplate 实现中, 每个send() 方法的重载版本都接受一个额外的CorrelationData对象.
当启用了发布者确认时,此对象会在3.1.4, “AmqpTemplate”的回调中返回.这允许发送者使用确认(ack或nack)来关联发送的消息.

发布者返回

当模板的mandatory 属性为true时,返回消息将由 Section 3.1.4, “AmqpTemplate”描述的回调来返回.

从1.4版本开始,RabbitTemplate 支持 SpEL mandatoryExpression 属性,它将对每个请求消息进行评估,作为根评估对象来解析成布尔值. Bean引用,如"@myBean.isMandatory(#root)" 可用在此表达式中.

发布者返回内部也可用于RabbitTemplate 的发送和接收操作中. 参考the section called “Reply Timeout” 来了解更多信息.

批量

从1.4.2版本开始,引入了BatchingRabbitTemplate.它是RabbitTemplate 的子类,覆盖了send 方法,此方法可根据BatchingStrategy来批量发送消息; 只有当一个批次完成时才会向RabbitMQ发送消息。

public interface BatchingStrategy {

	MessageBatch addToBatch(String exchange, String routingKey, Message message);

	Date nextRelease();

	Collection releaseBatches();

}
警告
成批的数据是保存在内存中的,如果出现系统故障,未发送的消息将会丢失.

这里提供了一个 SimpleBatchingStrategy .它支持将消息发送到单个 exchange/routing key.它有下面的属性:

  • batchSize - 发送前一个批次中消息的数量
  • bufferLimit - 批量消息的最大大小;如果超过了此值,它会取代batchSize, 并导致要发送的部分批处理
  • timeout - 当没有新的活动添加到消息批处理时之后,将发送部分批处理的时间(a time after which a partial batch will be sent when there is no new activity adding messages to the batch)

SimpleBatchingStrategy 通过在每个消息的前面嵌入4字节二进制长度来格式化批次消息. 这是通过设置springBatchFormat消息属性为lengthHeader4向接收系统传达的.

 

重要

批量消息自动由监听器容器来分批(de-batched)(使用springBatchFormat消息头).拒绝批量消息中的任何一个会将导致拒绝整个批次消息.

3.1.6 接收消息

介绍

Message 接收总是比发送稍显复杂.有两种方式来接收Message. 最简单的选择是在轮询方法调用中一次只接收一个消息. 更复杂的更常见的方法是注册一个侦听器,按需异步的接收消息。
在下面两个子章节中,我们将看到这两种方法的示例.

Polling Consumer

AmqpTemplate 自身可用来轮询消息接收.默认情况下,如果没有可用消息,将会立即返回 null;它是无阻塞的.
从1.5版本开始,你可以设置receiveTimeout,以毫秒为单位, receive方法会阻塞设定的时间来等待消息.小于0的值则意味着无限期阻塞 (或者至少要等到与broker的连接丢失). 
1.6版本引入了receive 方法的变种,以允许在每个调用上都可设置超时时间.

警告

由于接收操作会为每个消息创建一个新的QueueingConsumer,这种技术并不适用于大容量环境,可考虑使用异步消费者,或将receiveTimeout 设为0来应对这种情况.

这里有四个简单可用的receive 方法.同发送方的交换器一样, 有一种方法需要直接在模板本身上设置的默认队列属性, 还有一种方法需要在运行接受队列参数. 
版本1.6 引入了接受timeoutMillis 的变种,基于每个请求重写了receiveTimeout 方法.

Message receive() throws AmqpException;

Message receive(String queueName) throws AmqpException;

Message receive(long timeoutMillis) throws AmqpException;

Message receive(String queueName, long timeoutMillis) throws AmqpException;

与发送消息的情况类似, AmqpTemplate 有一些便利的方法来接收POJOs 而非Message 实例, 其实现可提供一种方法来定制MessageConverter 以用于创建返回的Object:

Object receiveAndConvert() throws AmqpException;

Object receiveAndConvert(String queueName) throws AmqpException;

Message receiveAndConvert(long timeoutMillis) throws AmqpException;

Message receiveAndConvert(String queueName, long timeoutMillis) throws AmqpException;

类似于sendAndReceive 方法,从1.3版本开始, AmqpTemplate 有多个便利的receiveAndReply 方法同步接收,处理,以及回应消息:

 boolean receiveAndReply(ReceiveAndReplyCallback callback)
	   throws AmqpException;

 boolean receiveAndReply(String queueName, ReceiveAndReplyCallback callback)
 	throws AmqpException;

 boolean receiveAndReply(ReceiveAndReplyCallback callback,
	String replyExchange, String replyRoutingKey) throws AmqpException;

 boolean receiveAndReply(String queueName, ReceiveAndReplyCallback callback,
	String replyExchange, String replyRoutingKey) throws AmqpException;

 boolean receiveAndReply(ReceiveAndReplyCallback callback,
 	ReplyToAddressCallback replyToAddressCallback) throws AmqpException;

 boolean receiveAndReply(String queueName, ReceiveAndReplyCallback callback,
			ReplyToAddressCallback replyToAddressCallback) throws AmqpException;

AmqpTemplate 实现会负责receive 和 reply 阶段.在大多数情况下,如果有必要,你只需要提供ReceiveAndReplyCallback 的实现来为收到的消息执行某些业务逻辑或为收到的消息构建回应对象.
注意,ReceiveAndReplyCallback 可能返回null. 在这种情况下,将不会发送回应,receiveAndReply 的工作类似于receive 方法. 这允许相同的队列用于消息的混合物,其中一些可能不需要答复。

自动消息(请求和应答)转换只能适应于提供的回调不是ReceiveAndReplyMessageCallback 实例的情况下- 它提供了一个原始的消息交换合同。

ReplyToAddressCallback 只在这种情况中有用,需要根据收到的信息通过自定义逻辑来决定replyTo 地址,并在ReceiveAndReplyCallback中进行回应的情况. 默认情况下,请求消息中的 replyTo 信息用来路由回复.

下面是一个基于POJO的接收和回复…​

boolean received =
        this.template.receiveAndReply(ROUTE, new ReceiveAndReplyCallback() {

                public Invoice handle(Order order) {
                        return processOrder(order);
                }
        });
if (received) {
        log.info("We received an order!");
}

异步消费者

重要 

Spring AMQP 也支持注解监听器endpoints(通过使用 @RabbitListener 注解)并提供了一个开放的基础设施,编程注册端点。
这是目前为止建立一个异步消费者的最方便方式, 参考the section called “Annotation-driven Listener Endpoints”来了解更多详情.

消息监听器

对于异步消息接收, 会涉及到一个专用组件(不是AmqpTemplate).此组件可作为消息消费回调的容器. 
稍后,我们会讲解这个容器和它的属性,但首先让我们来看一下回调,因为这里是你的应用程序代码与消息系统集成的地方. MessageListener 接口:

public interface MessageListener {
    void onMessage(Message message);
}

如果出于任何理由,你的回调逻辑需要依赖于AMQP Channel实例,那么你可以使用ChannelAwareMessageListener. 它看起来是很相似的,但多了一个额外的参数:

public interface ChannelAwareMessageListener {
    void onMessage(Message message, Channel channel) throws Exception;
}

MessageListenerAdapter

如果您希望在应用程序逻辑和消息API之间保持严格的分离,则可以依赖于框架所提供的适配器实现。

这是通常被称为“消息驱动的POJO”支持。当使用适配器时,只需要提供一个适配器本身应该调用的实例引用即可。

MessageListenerAdapter listener = new MessageListenerAdapter(somePojo);
    listener.setDefaultListenerMethod("myMethod");

你也可以继承适配器,并实现getListenerMethodName()方法(基于消息来动态选择不同的方法). 这个方法有两个参数:originalMessage 和extractedMessage, 后者是转换后的结果.默认情况下,需要配置SimpleMessageConverter ; 
参考the section called “SimpleMessageConverter” 来了解更多信息以及其它转换器的信息.

从1.4.2开始,原始消息包含consumerQueue 和 consumerTag 属性,这些属性可用来确定消息是从那个队列中收到的.

从1.5版本开始,你可以配置消费者queue/tag到方法名称的映射(map)以动态选择要调用的方法.如果map中无条目,我们将退回到默认监听器方法.

容器

你已经看过了消息监听回调上的各种各样的选项,现在我们将注意力转向容器. 基本上,容器处理主动(active)的职责,这样监听器回调可以保持被动(passive). 容器是“生命周期”组件的一个例子。
它提供了启动和停止的方法.当配置容器时,你本质上缩短了AMQP Queue和 MessageListener 实例之间的距离.你必须提供一个ConnectionFactory 的引用,队列名称或队列实例.
下面是使用默认实现SimpleMessageListenerContainer 的最基础的例子:

SimpleMessageListenerContainer container = new SimpleMessageListenerContainer();
container.setConnectionFactory(rabbitConnectionFactory);
container.setQueueNames("some.queue");
container.setMessageListener(new MessageListenerAdapter(somePojo));

作为一个主动组件, 最常见的是使用bean定义来创建监听器容器,这样它就可以简单地运行于后台.这可以通过XML来完成:

 
   

或者你可以@Configuration 风格:

@Configuration
public class ExampleAmqpConfiguration {

    @Bean
public SimpleMessageListenerContainer messageListenerContainer() {
        SimpleMessageListenerContainer container = new SimpleMessageListenerContainer();
        container.setConnectionFactory(rabbitConnectionFactory());
        container.setQueueName("some.queue");
        container.setMessageListener(exampleListener());
        return container;
    }

    @Bean
public ConnectionFactory rabbitConnectionFactory() {
        CachingConnectionFactory connectionFactory =
            new CachingConnectionFactory("localhost");
        connectionFactory.setUsername("guest");
        connectionFactory.setPassword("guest");
        return connectionFactory;
    }

    @Bean
public MessageListener exampleListener() {
        returnnew MessageListener() {
            publicvoid onMessage(Message message) {
                System.out.println("received: " + message);
            }
        };
    }
}

RabbitMQ Version 3.2开始, broker支持消费者优先级了(参考 Using Consumer Priorities with RabbitMQ). 

这可以通过在消费者设置x-priority 参数来启用. 

SimpleMessageListenerContainer 现在支持设置消费者参数:

container.setConsumerArguments(Collections.
 singletonMap("x-priority", Integer.valueOf(10)));

为了方便,命名空间在listener元素上提供了priority 属性:

 
   

从1.3版本开始,容器监听的队列可在运行时进行修改,参考 Section 3.1.18, “Listener Container Queues”.

auto-delete 队列

当容器配置为监听auto-delete 队列或队列有x-expires 选项或者broker配置了Time-To-Live 策略,队列将在容器停止时(最后的消费者退出时)由broker进行删除. 
在1.3版本之前,容器会因队列缺失而不能重启; 当连接关闭/打开时,RabbitAdmin 只能自动重新声明队列.

1.3版本开始, 在启动时,容器会使用RabbitAdmin 来重新声明缺失的队列.

您也可以使用条件声明(the section called “Conditional Declaration”) 与auto-startup="false" 来管理队列的延迟声明,直到容器启动.

 
   
 
   
 
   

在这种情况下,队列和交换器是由 containerAdmin 来声明的,auto-startup="false" 因此在上下文初始化期间不会声明元素.同样,出于同样原因,容器也不会启动.当容器随后启动时,它会使用containerAdmin引用来声明元素.

批量消息

批量消息会自动地通过监听器容器 (使用springBatchFormat 消息头)来解批(de-batched). 拒绝批量消息中的任何一个都将导致整批消息被拒绝. 参考the section called “Batching” 来了解更多关于批量消息的详情.

 

消费者失败事件

从1.5版本开始,无论时候,当监听器(消费者)经历某种失败时,SimpleMessageListenerContainer 会发布应用程序事件. 事件ListenerContainerConsumerFailedEvent 有下面的属性:

  • container - 消费者经历问题的监听容器.
  • reason - 失败的文本原因。
  • fatal - 一个表示失败是否是致命的boolean值;对于非致命异常,容器会根据retryInterval值尝试重新启动消费者.
  • throwable -捕捉到的Throwable.

这些事件能通过实现ApplicationListener来消费.

当 concurrentConsumers 大于1时,系统级事件(如连接失败)将发布到所有消费者.

如果消费者因队列是专有使用而失败了,默认情况下,在发布事件的时候,也会发出WARN 日志. 要改变日志行为,需要在SimpleMessageListenerContainer的exclusiveConsumerExceptionLogger属性中提供自定义的ConditionalExceptionLogger
也可参考the section called “Logging Channel Close Events”.

致命错误都记录在ERROR级别中,这是不可修改的。

 

-第四部分

Consumer Tags

从1.4.5版本开始,你可以提供一种策略来生成consumer tags.默认情况下,consumer tag是由broker来生成的.

public interface ConsumerTagStrategy {      String createConsumerTag(String queue);  }

该队列是可用的,所以它可以(可选)在tag中使用。

参考Section 3.1.15, “Message Listener Container Configuration”.

注解驱动的监听器Endpoints

介绍

从1.4版本开始,异步接收消息的最简单方式是使用注解监听器端点基础设施. 简而言之,它允许你暴露管理bean的方法来作为Rabbit 监听器端点.

@Component
public class MyService {      
 @RabbitListener(queues = "myQueue")
 public void processOrder(String data) {         
     ...     
  }  
}
上面例子的含义是,当消息在org.springframework.amqp.core.Queue "myQueue"上可用时, 会调用processOrder方法(在这种情况下,带有消息的负载).

通过使用RabbitListenerContainerFactory,注解端点基础设施在每个注解方法的幕后都创建了一个消息监听器容器.在上面的例子中,myQueue 必须是事先存在的,
并绑定了某个交换器上.从1.5.0版本开始,只要在上下文中存在RabbitAdmin,队列可自动声明和绑定.
 

@Component
public class MyService {

  @RabbitListener(bindings = @QueueBinding(
        value = @Queue(value = "myQueue", durable = "true"),
        exchange = @Exchange(value = "auto.exch", ignoreDeclarationExceptions = "true"),
        key = "orderRoutingKey")
  )public void processOrder(String data) {
    ...
  }

  @RabbitListener(bindings = @QueueBinding(
        value = @Queue,
        exchange = @Exchange(value = "auto.exch"),
        key = "invoiceRoutingKey")
  )public void processInvoice(String data) {
    ...
  }

}

在第一个例子中,队列myQueue 会与交换器一起自动声明(持久化的), 如果需要,可使用路由键来绑定到交换器上.在第二个例子中,匿名(专用的,自动删除的)队列将会声明并绑定. 
可提供多个 QueueBinding 条目,允许监听器监听多个队列.

当前只支持DIRECT, FANOUT, TOPIC 和HEADERS的交换器类型.当需要高级配置时,可使用@Bean 定义.

注意第一个例子中交换器上的 ignoreDeclarationExceptions .这允许,例如, 绑定到有不同的设置(如.internal)的交换器上. 默认情况下,现有交换器的属性必须被匹配.

从1.6版本开始,你可为队列,交换器和绑定的@QueueBinding 注解中指定参数.示例:
 

@RabbitListener(bindings = @QueueBinding(
        value = @Queue(value = "auto.headers", autoDelete = "true",
                        arguments = @Argument(name = "x-message-ttl", value = "10000",
                                                type = "java.lang.Integer")),
        exchange = @Exchange(value = "auto.headers", type = ExchangeTypes.HEADERS, autoDelete = "true"),
        arguments = {
                @Argument(name = "x-match", value = "all"),
                @Argument(name = "foo", value = "bar"),
                @Argument(name = "baz")
        })
)
public String handleWithHeadersExchange(String foo) {
    ...
}

注意队列的x-message-ttl 参数设为了10秒钟,因为参数类型不是String, 因此我们指定了它的类型,在这里是Integer.有了这些声明后,如果队列已经存在了,参数必须匹配现有队列上的参数.对于header交换器,我们设置binding arguments 要匹配头中foo为bar,且baz可为任意值的消息. x-match 参数则意味着必须同时满足两个条件.

参数名称,参数值,及类型可以是属性占位符(${...}) 或SpEL 表达式(#{...}). name 必须要能解析为String; type的表达式必须能解析为Class 或类的全限定名. value 必须能由DefaultConversionService 类型进行转换(如上面例子中x-message-ttl).

如果name 解析为null 或空字符串,那么将忽略 @Argument.

元注解(Meta-Annotations)

有时,你想将同样的配置用于多个监听器上. 为减少重复配置,你可以使用元注解来创建你自己的监听器注解:

@Target({ElementType.TYPE, ElementType.METHOD, ElementType.ANNOTATION_TYPE})
@Retention(RetentionPolicy.RUNTIME)
@RabbitListener(bindings = @QueueBinding(
        value = @Queue,
        exchange = @Exchange(value = "metaFanout", type = ExchangeTypes.FANOUT)))
public@interface MyAnonFanoutListener {
}

public class MetaListener {

    @MyAnonFanoutListener
public void handle1(String foo) {
        ...
    }

    @MyAnonFanoutListener
public void handle2(String foo) {
        ...
    }

}

在这个例子中,每个通过@MyAnonFanoutListener创建的监听器都会绑定一个匿名,自动删除的队列到fanout交换器 metaFanout上. 元注解机制是简单的,在那些用户定义注解中的属性是不会经过检查的- 因此你不能从元注解中覆盖设置.当需要高级配置时,使用一般的 @Bean 定义.

Enable Listener Endpoint Annotations

为了启用 @RabbitListener 注解,需要在你的某个@Configuration类中添加@EnableRabbit 注解.

@Configuration
@EnableRabbit
publicclass AppConfig {

    @Bean
public SimpleRabbitListenerContainerFactory rabbitListenerContainerFactory() {
        SimpleRabbitListenerContainerFactory factory = new SimpleRabbitListenerContainerFactory();
        factory.setConnectionFactory(connectionFactory());
        factory.setConcurrentConsumers(3);
        factory.setMaxConcurrentConsumers(10);
        return factory;
    }
}

默认情况下,基础设施会查找一个名为rabbitListenerContainerFactory 的bean作为工厂来源来创建消息监听器容器. 在这种情况下,会忽略RabbitMQ 基础设施计划, processOrder 方法可使用核心轮询大小为3个线程最大10个线程的池大小来调用.

可通过使用注解或实现RabbitListenerConfigurer

接口来自定义监听器容器工厂. 默认只需要注册至少一个Endpoints,而不需要一个特定的容器工厂.查看javadoc来了解详情和例子.

如果你更喜欢XML配置,可使用  元素.

 
   

注解方法的消息转换

在调用监听器之前,在管道中有两个转换步骤. 第一个使用 MessageConverter 来将传入的Spring AMQP Message 转换成spring-消息系统的消息. 当目标方法调用时,消息负载将被转换,如果有必要,也会参考消息参数类型来进行.

第一步中的默认 MessageConverter 是一个Spring AMQP SimpleMessageConverter ,它可以处理String 和 java.io.Serializable对象之间的转换; 其它所有的将保留为byte[]. 在下面的讨论中,我们称其为消息转换器.

第二个步骤的默认转换器是GenericMessageConverter ,它将委派给转换服务(DefaultFormattingConversionService的实例). 在下面的讨论中,我们称其为方法参数转换器.

要改变消息转换器,可在连接工厂bean中设置其相关属性:

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

这配置了一个Jackson2 转换器,希望头信息能通过它来指导转换.

你也可以考虑使用ContentTypeDelegatingMessageConverter ,它可以处理不同内容类型的转换.

大多数情况下,没有必要来定制方法参数转换器,除非你想要用自定义的ConversionService.

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

类型推断只能用于 @RabbitListener 的方法级.

参考 the section called “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 ConversionService myConversionService() {
    	DefaultConversionService conv = new DefaultConversionService();
    	conv.addConverter(mySpecialConverter());
    	return conv;
    }

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

    ...

}
重要
对于多方法监听器(参考 the section called “Multi-Method Listeners”), 方法选择是基于消息转换后的消息负载,方法参数转换器只在方法被选择后才会调用.

编程式 Endpoint 注册

RabbitListenerEndpoint 提供了一个Rabbit endpoint 模型并负责为那个模型配置容器.除了通过RabbitListener注解检测外,这个基础设施允许你通过编程来配置endpoints.

@Configuration
@EnableRabbit
publicclass AppConfig implements RabbitListenerConfigurer {

    @Override
publicvoid configureRabbitListeners(RabbitListenerEndpointRegistrar registrar) {
        SimpleRabbitListenerEndpoint endpoint = new SimpleRabbitListenerEndpoint();
        endpoint.setQueueNames("anotherQueue");
        endpoint.setMessageListener(message -> {
            // processing
        });
        registrar.registerEndpoint(endpoint);
    }
}

在上面的例子中,我们使用了SimpleRabbitListenerEndpoint (它使用MessageListener 来进行处理),但你也可以构建你自己的endpoint变种来描述自定义的调用机制.

应该指出的是,你也可以跳过@RabbitListener 的使用,通过RabbitListenerConfigurer来编程注册你的endpoints.

Annotated Endpoint Method Signature

到目前为止,我们已经在我们的端点上注入了一个简单的字符串,但它实际上可以有一个非常灵活的方法签名。让我们重写它,以一个自定义的头来控制注入顺序:

@Component
publicclass MyService {

    @RabbitListener(queues = "myQueue")
publicvoid processOrder(Order order, @Header("order_type") String orderType) {
        ...
    }
}

下面是你可以在监听端点上注入的主要元素:

原生org.springframework.amqp.core.Message.

用于接收消息的com.rabbitmq.client.Channel

 org.springframework.messaging.Message 代表的是传入的AMQP消息.注意,这个消息持有自定义和标准的头部信息 (AmqpHeaders定义).

 

从1.6版本开始, 入站deliveryMode 头可以AmqpHeaders.RECEIVED_DELIVERY_MODE 使用,代替了AmqpHeaders.DELIVERY_MODE.

@Header-注解方法参数可 提取一个特定头部值,包括标准的AMQP头.

@Headers-注解参数为了访问所有头信息,必须能指定为java.util.Map.

非注解元素(非支持类型(如. Message 和Channel))可认为是负荷(payload).你可以使用 @Payload来明确标识. 你也可以添加额外的 @Valid来进行验证.

注入Spring消息抽象的能力是特别有用的,它可受益于存储在特定传输消息中的信息,而不需要依赖于特定传输API.

@RabbitListener(queues = "myQueue")
public void processOrder(Message order) { ...
}

方法参数的处理是由DefaultMessageHandlerMethodFactory 提供的,它可以更进一步地定制以支持其它的方法参数. 转换和验证支持也可以定制.

例如,如果我们想确保我们的Order在处理之前是有效的,我们可以使用@Valid 来注解负荷,并配置必须验证器,就像下面这样:

@Configuration
@EnableRabbit
public class AppConfig implements RabbitListenerConfigurer {

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

    @Bean
public DefaultMessageHandlerMethodFactory myHandlerMethodFactory() {
        DefaultMessageHandlerMethodFactory factory = new DefaultMessageHandlerMethodFactory();
        factory.setValidator(myValidator());
        return factory;
    }
}

监听多个队列

当使用queues 属性时,你可以指定相关的容器来监听多个队列. 你可以使用 @Header 注解来指定对于那些队列中收到的消息对POJO方法可用:

@Component
public class MyService {

    @RabbitListener(queues = { "queue1", "queue2" } )
public void processOrder(String data, @Header(AmqpHeaders.CONSUMER_QUEUE) String queue) {
        ...
    }

}

从1.5版本开始,队列名称可以使用属性占位符和SpEL:

@Component
public class MyService {

    @RabbitListener(queues = "#{'${property.with.comma.delimited.queue.names}'.split(',')}" )
public void processOrder(String data, @Header(AmqpHeaders.CONSUMER_QUEUE) String queue) {
        ...
    }

}

1.5版本之前,只有单个队列可以这种方法进行指定,每个队列需要一个单独的属性.

回复管理

MessageListenerAdapter 现有的支持已经允许你的方法有一个非void的返回类型.在这种情况下,调用的结果被封装在一个发送消息中,其消息发送地址要么是原始消息的ReplyToAddress头指定的地址要么是监听器上配置的默认地址.默认地址现在可通过@SendTo 注解进行设置.

假设我们的processOrder 方法现在需要返回一个OrderStatus, 可将其写成下面这样来自动发送一个回复:

@RabbitListener(destination = "myQueue")
@SendTo("status")
public OrderStatus processOrder(Order order) {
    // order processing
 return status;
}

如果你需要以传输独立的方式来设置其它头,你可以返回Message,就像这样:

@RabbitListener(destination = "myQueue")
@SendTo("status")
public Message processOrder(Order order) {
    // order processing
return MessageBuilder
        .withPayload(status)
        .setHeader("code", 1234)
        .build();
}

@SendTo 值按照exchange/routingKey模式(其中的一部分可以省略)来作为对exchange 和 routingKey 的回复.有效值为:

foo/bar - 以交换器和路由键进行回复.

foo/ - 以交换器和默认路由键进行回复.

bar or /bar - 以路由键和默认交换器进行回复.

/ or empty - 以默认交换器和默认路由键进行回复.

 @SendTo 也可以没有value 属性. 这种情况等价于空的sendTo 模式. @SendTo 只能应用于没有replyToAddress 属性的入站消息中.

从1.5版本开始, @SendTo 值可以通过bean SpEL 表达式初始化,例如…​

@RabbitListener(queues = "test.sendTo.spel")
@SendTo("#{spelReplyTo}")
public String capitalizeWithSendToSpel(String foo) {
    return foo.toUpperCase();
}
...
@Bean
public String spelReplyTo() {
    return"test.sendTo.reply.spel";
}

表达式必须能评估为String,它可以是简单的队列名称(将发送到默认交换器中) 或者是上面谈到的exchange/routingKey 形式.

在初始化时,#{...} 表达式只评估一次.

对于动态路由回复,消息发送者应该包含一个reply_to 消息属性或使用运行时SpEL 表达式.

从1.6版本开始, @SendTo 可以是SpEL 表达式,它可在运行时根据请求和回复来评估:

@RabbitListener(queues = "test.sendTo.spel")
@SendTo("!{'some.reply.queue.with.' + result.queueName}")
public Bar capitalizeWithSendToSpel(Foo foo) {
    return processTheFooAndReturnABar(foo);
}

SpEL 表达式的运行时性质是由 !{...} 定界符表示的. 表达式评估上下文的#root 对象有三个属性:

  • request - o.s.amqp.core.Message 请求对象.
  • source - 转换后的 o.s.messaging.Message.
  • result - 方法结果.

上下文有一个map 属性访问器,标准类型转换器以及一个bean解析器,允许引用上下文中的其它beans (如.@someBeanName.determineReplyQ(request, result)).

总结一下, #{...} 只在初始化的时候评估一次, #root 对象代表的是应用程序上下文; beans可通过其名称来引用. !{...} 会在运行时,对于每个消息,都将使用root对象的属性进行评估,bean可以使用其名称进行引用,前辍为@.

多方法监听器

从1.5.0版本开始,@RabbitListener 注解现在可以在类级上进行指定.与新的@RabbitHandler 注解一起,基于传入消息的负荷类型,这可以允许在单个监听器上调用不同的方法.这可以用一个例子来描述:

@RabbitListener(id="multi", queues = "someQueue")
publicclass MultiListenerBean {

    @RabbitHandler
@SendTo("my.reply.queue")
public String bar(Bar bar) {
        ...
    }

    @RabbitHandler
public String baz(Baz baz) {
        ...
    }

    @RabbitHandler
public String qux(@Header("amqp_receivedRoutingKey") String rk, @Payload Qux qux) {
        ...
    }

}

在这种情况下,独立的 @RabbitHandler 方法会被调用,如果转换后负荷是BarBaz 或Qux. 理解基于负荷类型系统来确定唯一方法是很重要的.类型检查是通过单个无注解参数来执行的,否则就要使用@Payload 进行注解. 注意同样的方法签名可应用于方法级 @RabbitListener 之上.

注意,如果有必要,需要在每个方法上指定@SendTo, 在类级上它是不支持的.

@Repeatable @RabbitListener

从1.6版本开始,@RabbitListener 注解可用 @Repeatable进行标记. 这就是说,这个注解可多次出现在相同的注解元素上(方法或类).在这种情况下,对于每个注解,都会创建独立的监听容器,它们每个都会调用相同的监听器@Bean. Repeatable 注解能用于 Java 8+;当在Java 7-使用时,同样的效果可以使用 @RabbitListeners "container" 注解(包含@RabbitListener注解的数组)来达到.

Proxy @RabbitListener and Generics

如果你的服务是用于代理(如,在 @Transactional的情况中) ,当接口有泛型参数时,需要要一些考虑.要有一个泛型接口和特定实现,如:

interface TxService

{ String handle(P payload, String header); } static class TxServiceImpl implements TxService { @Override

 @RabbitListener(...)
 public String handle(Foo foo, String rk) {
         ...
    }

}

你被迫切换到CGLIB目标类代理,因为接口handle方法的实际实现只是一个桥接方法.在事务管理的情况下, CGLIB是通过注解选项来配置的- @EnableTransactionManagement(proxyTargetClass = true). 在这种情况下,所有注解都需要在实现类的目标方法上进行声明:

static class TxServiceImpl implements TxService {

 @Override
@Transactional
@RabbitListener(...)
public String handle(@Payload Foo foo, @Header("amqp_receivedRoutingKey") String rk) {
        ...
    }

}

容器管理

由注解创建的容器不会在上下文中进行注册.你可以调用 RabbitListenerEndpointRegistry的getListenerContainers()方法来获取所有容器集合.然后,你可以迭代这个集合,例如,停止/启动所有容器或调用在其注册上调用Lifecycle 方法(调用每个容器中的操作).

你也可以使用id来获取单个容器的引用,即 getListenerContainer(String id); 例如registry.getListenerContainer("multi") .

从1.5.2版本开始,你可以调用getListenerContainerIds()方法来获取所有注册容器的id.

从1.5版本开始,你可在RabbitListener端点上为容器分配一个组(group).这提供了一种机制来获取子集容器的引用; 添加一个group 属性会使Collection 类型的bean使用组名称注册在上下文中.

 

线程和异步消费者

一些不同的线程可与异步消费者关联。

当RabbitMQ Client投递消息时,来自于SimpleMessageListener 配置的TaskExecutor中的线程会调用MessageListener.如果没有配置,将会使用SimpleAsyncTaskExecutor. 如果使用了池化的executor,须确保池大小可以支撑并发处理.

当使用默认SimpleAsyncTaskExecutor时,对于调用监听器的线程,监听器容器的beanName 将用作threadNamePrefix. 这有益于日志分析,在日志appender配置中,一般建议总是包含线程名称.当在SimpleMessageListenerContainer的taskExecutor属性中指定TaskExecutor 时,线程名称是不能修改的.建议你使用相似的技术来命名线程, 帮助在日志消息中的线程识别。

当创建连接时,在CachingConnectionFactory 配置的Executor将传递给RabbitMQ Client ,并且它的线程将用于投递新消息到监听器容器.在写作的时候,如果没有配置,client会使用池大小为5的内部线程池executor.

RabbitMQ client 使用ThreadFactory 来为低端I/O(socket)操作创建线程.要改变这个工厂,你需要配置底层RabbitMQ ConnectionFactory, 正如the section called “Configuring the Underlying Client Connection Factory”中所描述.

 

检测空闲异步消费者

虽然高效,但异步消费者存在一个问题:如何来探测它们什么是空闲的 - 当有一段时间没有收到消息时,用户可能想要采取某些动作.

从1.6版本开始, 当没有消息投递时,可配置监听器容器来发布ListenerContainerIdleEvent 事件. 当容器是空闲的,事件会每隔idleEventInterval 毫秒发布事件.

要配置这个功能,须在容器上设置idleEventInterval:

xml

 
   

Java

@Bean
public SimpleMessageListenerContainer(ConnectionFactory connectionFactory) {
    SimpleMessageListenerContainer container = new SimpleMessageListenerContainer(connectionFactory);
    ...
    container.setIdleEventInterval(60000L);
    ...
    return container;
}

@RabbitListener

@Bean
public SimpleRabbitListenerContainerFactory rabbitListenerContainerFactory() {
    SimpleRabbitListenerContainerFactory factory = new SimpleRabbitListenerContainerFactory();
    factory.setConnectionFactory(rabbitConnectionFactory());
    factory.setIdleEventInterval(60000L);
    ...
    return factory;
}

在上面这些情况中,当容器空闲时,每隔60秒就会发布事件.

事件消费

通过实现ApplicationListener 可捕获这些事件- 要么是一个一般的监听器,要么是一个窄化的只接受特定事件的监听器. 你也可以使用Spring Framework 4.2中引入的@EventListener.

下面的例子在单个类中组合使用了@RabbitListener 和@EventListener .重点要理解,应用程序监听器会收到所有容器的事件,因此如果你只对某个容器采取措施,那么你需要检查监听器id.你也可以使用@EventListener 条件来达到此目的.

事件有4个属性:

  • source - 监听容器实例
  • id - 监听器id(或容器bean名称)
  • idleTime - 当事件发布时,容器已经空闲的时间
  • queueNames - 容器监听的队列名称
public class Listener {

    @RabbitListener(id="foo", queues="#{queue.name}")
    public String listen(String foo) {
        return foo.toUpperCase();
    }

    @EventListener(condition = "event.listenerId == 'foo'")
    public void onApplicationEvent(ListenerContainerIdleEvent event) {
        ...
    }

}
重要
事件监听器会查看所有容器的事件,因此,在上面的例子中,我们根据监听器ID缩小了要接收的事件.
警告
如果你想使用idle事件来停止监听器容器,你不应该在调用监听器的线程上来调用container.stop() 方法- 它会导致延迟和不必要的日志消息. 相反,你应该把事件交给一个不同的线程,然后可以停止容器。

3.1.7 消息转换器

介绍

AmqpTemplate 同时也定义了多个发送和接收消息(委派给MessageConverter)的方法.

MessageConverter 本身是很简单的. 在每个方向上它都提供了一个方法:一个用于转换成Message,另一个用于从Message中转换.注意,当转换成Message时,除了object外,你还需要提供消息属性. "object"参数通常对应的是Message body.

public interface MessageConverter {

    Message toMessage(Object object, MessageProperties messageProperties)
            throws MessageConversionException;

    Object fromMessage(Message message) throws MessageConversionException;

}

AmqpTemplate中相关的消息发送方法列举在下边. 这比我们前面提到的要简单,因为它们不需要Message 实例. 相反地,  MessageConverter 负责创建每个消息(通过将提供的对象转换成Message body的字节数组,以及添加提供的MessageProperties).

void convertAndSend(Object message) throws AmqpException;

void convertAndSend(String routingKey, Object message) throws AmqpException;

void convertAndSend(String exchange, String routingKey, Object message)
    throws AmqpException;

void convertAndSend(Object message, MessagePostProcessor messagePostProcessor)
    throws AmqpException;

void convertAndSend(String routingKey, Object message,
    MessagePostProcessor messagePostProcessor) throws AmqpException;

void convertAndSend(String exchange, String routingKey, Object message,
    MessagePostProcessor messagePostProcessor) throws AmqpException;

在接收端,这里只有两个方法:一个接受队列名称,另一个依赖于模板设置的队列属性.

Object receiveAndConvert() throws AmqpException;

Object receiveAndConvert(String queueName) throws AmqpException;
在 the section called “Asynchronous Consumer” 中提到的MessageListenerAdapter也使用了MessageConverter.

SimpleMessageConverter

MessageConverter 策略的默认实现被称为SimpleMessageConverter. 如果你没有明确配置,RabbitTemplate实例会使用此转换器的实例.它能处理基于文本内容,序列化Java对象,以及简单的字节数组.

从 Message中转换

如果传入消息的内容类型以"text" (如. "text/plain")开头,它同时也会检查内容编码属性,以确定将消息body字节数组转换成字符串所要使用的字符集. 如果在输入消息中没有指定内容编码属性, 它默认会使用"UTF-8"字符集.如果你需要覆盖默认设置,你可以配置一个SimpleMessageConverter实例,设置其"defaultCharset" 属性,再将其注入到RabbitTemplate 实例中.

如果传入消息的内容类型属性值为"application/x-java-serialized-object", SimpleMessageConverter 将尝试将字节数组反序列化为一个Java object. 虽然这对于简单的原型是有用的,但一般不推荐依赖于Java序列化机制,因为它会生产者和消费者之间的紧密耦合。当然,这也排除了在两边使用非Java的可能性.由于AMQP 是线路级协议, 因这样的限制失去了许多优势,这是不幸的. 在后面的两个章节中,我们将探讨通过丰富的域对象的内容来替代java序列化.

对于其它内容类型,SimpleMessageConverter 会以字节数组形式直接返回消息body内容.

参考the section called “Java Deserialization” 来了解更多信息.

转换成消息

当从任意Java对象转换成Message时, SimpleMessageConverter 同样可以处理字节数组,字符串,以及序列化实例.它会将每一种都转换成字节(在字节数组的情况下,不需要任何转换), 并且会相应地设置内容类型属性.如果要转换的对象不匹配这些类型,Message body 将是null.

SerializerMessageConverter

除了它可以使用其它application/x-java-serialized-object转换的Spring框架Serializer 和 Deserializer 实现来配置外,此转换器类似于SimpleMessageConverter

参考the section called “Java Deserialization” 来了解更多信息.

Jackson2JsonMessageConverter

转换成消息

正如前面章节提到的,一般来说依赖于Java序列化机制不是推荐的.另一个常见更灵活且可跨语言平台的选择JSON (JavaScript Object Notation).可通过在RabbitTemplate实例上配置转换器来覆盖默认SimpleMessageConverter.Jackson2JsonMessageConverter 使用的是com.fasterxml.jackson 2.x 包.

 
   
 
   
 
   

正如上面展示的, Jackson2JsonMessageConverter 默认使用的是DefaultClassMapper. 类型信息是添加到MessageProperties中的(也会从中获取). 如果入站消息在MessageProperties中没有包含类型信息,但你知道预期类型,你可以使用defaultType 属性来配置静态类型

 
   
 
   
 
   

转换Message

入站消息会根据发送系统头部中添加的类型信息来转换成对象.

在1.6之前的版本中,如果不存在类型信息,转换将失败。从1.6版开始,如果类型信息丢失,转换器将使用Jsckson默认值(通常是一个map)来转换JSON.

此外,从1.6版本开始,当在方法上使用@RabbitListener 注解时, 推断类型信息会添加到MessageProperties; 这允许转换器转换成目标方法的参数类型.这只适用于无注解的参数或使用@Payload注解的单个参数. 在分析过程中忽略类型消息的参数。
 

重要

默认情况下,推断类型信息会覆盖inbound __TypeId__ 和发送系统创建的相关headers. 这允许允许接收系统自动转换成不同的领域对象. 这只适用于具体的参数类型(不是抽象的或不是接口)或者来自java.util 包中的对象.其它情况下,将使用 __TypeId__ 和相关的头.也可能有你想覆盖默认行为以及总是使用__TypeId__信息的情况. 例如, 让我们假设你有一个接受Foo参数的@RabbitListener ,但消息中包含了Bar( 它是的Foo (具体类)的子类). 推断类型是不正确的.要处理这种情况,需要设置Jackson2JsonMessageConverter 的TypePrecedence 属性为TYPE_ID 而替换默认的INFERRED. 这个属性实际上转换器的DefaultJackson2JavaTypeMapper ,但为了方便在转换器上提供了一个setter方法. 如果你想注入一个自定义类型mapper, 你应该设置属性mapper.

@RabbitListener
public void foo(Foo foo) {...}

@RabbitListener
public void foo(@Payload Foo foo, @Header("amqp_consumerQueue") String queue) {...}

@RabbitListener
public void foo(Foo foo, o.s.amqp.core.Message message) {...}

@RabbitListener
public void foo(Foo foo, o.s.messaging.Message message) {...}

@RabbitListener
public void foo(Foo foo, String bar) {...}

@RabbitListener
public void foo(Foo foo, o.s.messaging.Message message) {...}

上面前4种情况下,转换器会尝试转换成Foo 类型. 第五个例子是无效的,因为我们不能确定使用哪个参数来接收消息负荷. 在第六个例子中, Jackson 会根据泛型WildcardType来应用.

然而,你也可以创建一个自定义转换器,并使用targetMethod 消息属性来决定将JSON转换成哪种类型.

这种类型接口只能在@RabbitListener 注解声明在方法级上才可实现.在类级@RabbitListener, 转换类型用来选择调用哪个@RabbitHandler 方法.基于这个原因,基础设施提供了targetObject 消息属性,它可用于自定义转换器来确定类型.

MarshallingMessageConverter

还有一个选择是MarshallingMessageConverter.它会委派到Spring OXM 包的 Marshaller 和 Unmarshaller 策略接口实现. 

你可从here了解更多. 在配置方面,最常见的是只提供构造器参数,因为大部分Marshaller 的实现都将实现Unmarshaller.

 
   
 
   
 
    
   

ContentTypeDelegatingMessageConverter

这个类是在1.4.2版本中引入的,并可基于MessageProperties的contentType属性允许委派给一个特定的MessageConverter.默认情况下,如果没有contentType属性或值没有匹配配置转换器时,它会委派给SimpleMessageConverter.

 
 
   
 
    
   

Java 反序列化

重要

当从不可信任的来源反序列化Java对象时,存在一个可能的漏洞.如果从不可信来源,使用内容类型 application/x-java-serialized-object来接收消息,你可以考虑配置允许哪些包/类能反序列化.这既适用于SimpleMessageConverter,也适用于SerializerMessageConverter,当它被配置为使用一个DefaultDeserializer时 -或含蓄地或通过配置方式的。

默认情况下,白名单列表是空的,这意味着所有类都会反序列化.你可以设置模式列表,如 foo.*foo.bar.Baz 或 *.MySafeClass.模式会按照顺序进行检查,直到找到匹配的模式.如果没有找到匹配,将抛出SecurityException.在这些转换器上,可使用whiteListPatterns 属性来设置.

消息属性转换器

 MessagePropertiesConverter 策略接口用于Rabbit Client BasicProperties 与Spring AMQP MessageProperties之间转换. 默认实现(DefaultMessagePropertiesConverter)通常可满虽大部分需求,但如果有需要,你可以自己实现. 当大小不超过1024字节时,默认属性转换器将 BasicProperties 中的LongString 转换成String . 更大的 LongString 将不会进行转换(参考下面的内容.这个限制可通过构造器参数来覆盖.

从1.6版本开始, 现在headers 长超过 long string 限制(默认为1024) 将被DefaultMessagePropertiesConverter保留作为 LongString . 你可以通过 the getBytes[]toString(), 或getStream() 方法来访问内容.

此前, DefaultMessagePropertiesConverter 会将这样的头转换成一个 DataInputStream (实际上它只是引用了LongString的DataInputStream). 在输出时,这个头不会进行转换(除字符串外,如在流上调用toString()方法 java.io.DataInputStream@1d057a39).

更大输入LongString 头现在可正确地转换,在输出时也一样.

它提供了一个新的构造器来配置转换器,这样可像以前一样来工作:

/**
 * Construct an instance where LongStrings will be returned
 * unconverted or as a java.io.DataInputStream when longer than this limit.
 * Use this constructor with 'true' to restore pre-1.6 behavior.
 * @param longStringLimit the limit.
 * @param convertLongLongStrings LongString when false,
 * DataInputStream when true.
 * @since 1.6
 */
public DefaultMessagePropertiesConverter(int longStringLimit, boolean convertLongLongStrings) { ... }

另外,从1.6版本开始,在 MessageProperties中添加了一个新属性correlationIdString.此前,当在RabbitMQ 客户端中转换BasicProperties 时,将会执行不必要的byte[] <-> String 转换,这是因为 MessageProperties.correlationId 是一个byte[] 而 BasicProperties 使用的是String

(最终,RabbitMQ客户端使用UTF-8字符串转化为字节并放在协议消息中).

 

为提供最大向后兼容性,新属性correlationIdPolicy 已经被加入到了DefaultMessagePropertiesConverter.它接受DefaultMessagePropertiesConverter.CorrelationIdPolicy 枚举参数.

默认情况下,它设置为BYTES (复制先前的行为).

对于入站消息:

  • STRING - 只映射correlationIdString 属性
  • BYTES - 只映射correlationId 属性
  • BOTH - 会同时映射两个属性

对于出站消息:

  • STRING - 只映射correlationIdString 属性
  • BYTES - 只映射correlationId 属性
  • BOTH - 两种属性都会考虑,但会优先考虑String 属性

也从1.6版本开始,入站deliveryMode 属性不再需要映射 MessageProperties.deliveryMode,相反使用MessageProperties.receivedDeliveryMode 来代替.另外,入站userId 属性也不需要再映射MessageProperties.userId,相反使用MessageProperties.receivedUserId 来映射. 

这种变化是为了避免这些属性的意外传播,如果同样的MessageProperties 对象用于出站消息时.

3.1.8 修改消息- 压缩以及更多

提供了许多的扩展点,通过它们你可以对消息执行预处理,要么在发送RabbitMQ之前,要么在接收到消息之后.

正如你在Section 3.1.7, “Message Converters”看到的,这样的扩展点存在于AmqpTemplate convertAndReceive 操作中,在那里你可以提供一个MessagePostProcessor

例如,你的POJO转换之后, MessagePostProcessor 允许你在Message上设置自定义的头或属性.

从1.4.2版本开始,额外的扩展点已经添加到RabbitTemplate - setBeforePublishPostProcessors() 和setAfterReceivePostProcessors(). 第一个开启了一个post processor来在发送消息到RabbitMQ之前立即运行.当使用批量时(参考 the section called “Batching”), 这会在批处理装配之后发送之前调用. 

第二个会在收到消息后立即调用.

这些扩展点对于压缩这此功能是有用的,基于这些目的,提供了多个MessagePostProcessor:

  • GZipPostProcessor
  • ZipPostProcessor

针对于发送前的消息压缩,以及

  • GUnzipPostProcessor
  • UnzipPostProcessor

针对于消息解压.

类似地, SimpleMessageListenerContainer 也有一个 setAfterReceivePostProcessors() 方法,

允许在消息收到由容器来执行解压缩.

 

-第五部分

3.1.9 Request/Reply 消息

介绍

AmqpTemplate 也提供了各种各样的sendAndReceive 方法,它们接受同样的参数选项(exchange, routingKey, and Message)来执行单向发送操作. 
这些方法对于request/reply 场景也是有用的,因为它们在发送前处理了必要的"reply-to"属性配置,并能通过它在专用队列(基于回复功能临时创建的队列)上监听回复消息.

类似的request/reply方法也是可用的, MessageConverter 可应用于请求和回复上.这些方法被称为convertSendAndReceive.参考 AmqpTemplate 的JavaDoc来了解详情.

从1.5.0版本开始,每个sendAndReceive 方法变种都有一个接受CorrelationData的重载版本. 连同正确配置的连接工厂,这使得发布者可以确认发送方的操作.
参考 the section called “Publisher Confirms and Returns” 来了解详情.

Reply 超时

默认情况下,这些 send和receive方法会在5秒后超时并返回null. 这可以通过设置replyTimeout属性来修改.
从1.5版本开始,如果你设置了 mandatory 属性为true (或特定消息上的 mandatory-expression 评估为true),如果消息不能投递到队列中,将抛出AmqpMessageReturnedException
这个 exception 有 returnedMessagereplyCodereplyText 属性, 如同用于发送的exchangeroutingKey

 

这个功能使用了发布者返回特性,可通过在CachingConnectionFactory上设置publisherReturns 为true来启用(参考the section called “Publisher Confirms and Returns”). 
此外,你不必在RabbitTemplate上注册你自己的ReturnCallback.

RabbitMQ Direct reply-to

重要

从3.4.0版本开始,RabbitMQ 服务器现在支持 Direct reply-to,基于主要原因,它消除了固定回复队列(为了避免为每个请求创建临时队列).
Spring AMQP 1.4.1 版本开始,Direct reply-to 就已经做为了默认使用(如果服务器支持的话),而不再创建临时队列.
当没有提供replyQueue(或设置名称为amq.rabbitmq.reply-to), RabbitTemplate 会自动探测是否支持Direct reply-to, 要么使用它或使用临时回复队列来回退. 当使用Direct reply-to, reply-listener不是必需的,不应该被配置。

Reply 监听器仍然运行命名队列(不是amq.rabbitmq.reply-to),允许控制并发回复.

从.16版本开始,出于某些原因,你想为每个回复使用临时的,专用的,自动删除的队列,你可以设置useTemporaryReplyQueues 属性为true. 如果你设置了replyAddress,此属性会被忽略.

决定是否使用 direct reply-to,可以通过继承RabbitTemplate 并覆盖useDirectReplyTo()来修改. 

此方法只在发出第一个请求时,调用一次.

应答队列的消息相关性

当使用固定回复队列时(不是amq.rabbitmq.reply-to), 必须要提供 correlation data,这样回复才能关联请求.参考RabbitMQ Remote Procedure Call (RPC). 
默认情况下,标准correlationId 属性会用来持有correlation data. 然而,如果你想使用自定义属性来持有correlation data, 你可在 中设置 correlation-key 属性.
显示设置属性为correlationId 将与缺省属性相同. 当然,客户端和服务器对于correlation data必须要有相同的头.

Spring AMQP 1.1版本为这个data使用自定义属性spring_reply_correlation.如果你想在当前版本中恢复这种行为,也许是为了保持1.1中的另一个应用程序的兼容性,你必须设置属性以spring_reply_correlation。

回复监听器容器

当使用3.4.0之前的Rabbit版本,每个回复都会使用一个新临时队列. 然而,可在模板上配置单个回复队列, 这将更加高效,同时也允许你在队列上设置参数.然而,在这种情况下,你必须提供子元素. 

这个元素为回复队列提供了监听器容器, 以模板为监听器. 
所有 Section 3.1.15, “Message Listener Container Configuration” 中的属性都可以配置在 元素中,除了connection-factory 和 message-converter(它们是模块配置中继承下来的).

重要

如果运行了多个应用程序实例或者使用了多个RabbitTemplate,那么你必须为每个都使用唯一的回复队列- RabbitMQ 没有在队列中选择消息的能力,如果它们都使用相同队列,每个实例都将竞争的答复,而不一定是收到他们自己的。

 
   
reply-queue="replies" reply-address="replyEx/routeReply">
 
   

由于容器和模板可共享一个连接工厂,它们不会共享一个通道,因此请求和回复不是在同一个事务中执行的(如果是事务的).

重要

在1.5.0版本之前, reply-address 属性不可用,回复总是通过默认交换器和reply-queue作路由键来进行的. 现在这依然是默认的,但现在你可以指定新的reply-address 属性. 
reply-address 可以包含/ 形式的地址,回复将会路由到设定的exchange和路由到routing key绑定的队列上. 
reply-address 优先于 reply-queue 必须配置为一个单独的 组件, 当只使用reply-address 时,无论是reply-address 还是 reply-queue (在中的queue属性) 必须指的是同一个队列.

在这个配置中,SimpleListenerContainer 用于接收回复; 而RabbitTemplate 将成为MessageListener. 当使用 命名空间元素定义模板时, 正如上面所展示的, 分析器会定义容器,并将模板作为监听器进行包装.

重要

当模板不使用固定 replyQueue (或使用Direct reply-to - 参考 the section called “RabbitMQ Direct reply-to”) ,则不需要监听器容器. 当在RabbitMQ3.4.0+使用时,Direct reply-to 是更好的机制.

 

如果你将 RabbitTemplate 定义为 , 或使用 @Configuration 类将其定义为@Bean,或者通过编程来创建模板,你需要自己定义和包装回复监听器容器.
如果这样做失败了,模板将不会收到回复,并最终会超时并返回null作为对sendAndReceive 方法调用者的回复.

从1.5版本开始, RabbitTemplate 会探测是否配置了MessageListener 来接收回复.如果没有,它会尝试发送并使用回复地地址来接收消息,如果失败了,则会抛出 IllegalStateException (因为不会收到回复).

此外,如果使用了简单的replyAddress (队列名称),回复监听器容器会验证与监听的队列是否是一样的名称.但如果这个地址是交换器和路由键,这种检查不会被执行,会输出调试日志信息.

重要

当在编写回复监听器和模板时,重要的一点是要保证模板的replyQueue 与容器的queues (或queueNames) 属性指的是相同的队列. 模板会将回复队列插入到出站消息的replyTo属性中.

下面的例子展示了如何来包装这些beans.

 
   
 
   
@Bean
public RabbitTemplate amqpTemplate() {         
RabbitTemplate rabbitTemplate = new RabbitTemplate(connectionFactory());         
rabbitTemplate.setMessageConverter(msgConv());         
rabbitTemplate.setReplyQueue(replyQueue());         
rabbitTemplate.setReplyTimeout(60000);         
return rabbitTemplate;     }      
@Bean
public SimpleMessageListenerContainer replyListenerContainer() {         
SimpleMessageListenerContainer container = new SimpleMessageListenerContainer();         
container.setConnectionFactory(connectionFactory());         
container.setQueues(replyQueue());         
container.setMessageListener(amqpTemplate());         
return container;     
}      
@Bean
public Queue replyQueue() {         
return new Queue("my.reply.queue");     
}

完整的RabbitTemplate 包装固定回复队列,与远程监听器容器的请求回复处理展示在 this test case.

重要

当回复超时时(replyTimeout), sendAndReceive() 方法会返回null.

 

在1.3.6版本之前, 消息超时回复只是简单地记录下来.现在,如果收到了迟到回复,将会拒绝(模板会抛出AmqpRejectAndDontRequeueException). 
如果回复队列配置了将拒绝消息到死信交换器中, 可获取回复来作后面的分析. 只须将队列以队列名称作为路由键绑定到死信交换器中.

参考RabbitMQ Dead Letter Documentation 来了解更多关于死信的配置信息.

你也可以看示例中关于FixedReplyQueueDeadLetterTests 测试用例.

AsyncRabbitTemplate

1.6版本引入了 AsyncRabbitTemplate

它有与 AmqpTemplate 上类似的sendAndReceive (和 convertSendAndReceive) 方法,但不是阻塞的,它们会返回一个 ListenableFuture.

sendAndReceive 方法返回一个RabbitMessageFutureconvertSendAndReceive 方法会返回一个RabbitConverterFuture.

你可以同步稍后在future上调用get()方法来获取结果,也可以注册一个回调异步来获取结果.

@Autowired
private AsyncRabbitTemplate template;  
...  
public void doSomeWorkAndGetResultLater() {      
...      
ListenableFuture future = this.template.convertSendAndReceive("foo");      
// do some more work      
String reply = null;     
try {         
reply = future.get();     
}     
catch (ExecutionException e) { 
        ...     
}      ...  
}  
public void doSomeWorkAndGetResultAsync() {      
...      
RabbitConverterFuture future = this.template.convertSendAndReceive("foo");     
future.addCallback(new ListenableFutureCallback() {          
@Override
publicvoid onSuccess(String result) { 
            ...         
}          
@Override
publicvoid onFailure(Throwable ex) { 
            ...         
}      
});      
...  
}

如果设置了mandatory ,且消息不能投递,future 会抛出一个ExecutionException ,并带有AmqpMessageReturnedException 原因,它封装了返回的消息和以及关于返回的信息.

如果设置了enableConfirms ,future会包含一个属性confirm ,它是 ListenableFuture , true 表示成功的发布.

如果confirm future是false,RabbitFuture 会有一个属性nackCause - 如果可用的话,则代表的是失败的原因.

重要

发布者确认已被废弃了(如果在回复之后收到),-因为回复已经暗示了成功发布.

在模板上设置receiveTimeout 属性来表示回复超时时间(它默认为 30 秒).如果发生了超时,future会以AmqpReplyTimeoutException结束.

模板可实现SmartLifecycle; 这样可阻止模板在等待回复时Future 退出.

Spring 远程调用 AMQP

Spring Framework 有一个普遍的远程处理能力, 允许 Remote Procedure Calls (RPC) 使用多种传输协议. Spring-AMQP 通过在客户端使用AmqpProxyFactoryBean ,在服务端使用AmqpInvokerServiceExporter也可以提供类似的机制. 
它提供了基于AMQP的RPC. 在客户端,RabbitTemplate 可以按照上面一样来使用,在服务器端, invoker (配置为MessageListener) 会收到消息, 调用配置的服务,使用入站消息的replyTo信息来返回回复.

客户端工厂可注入任何bean (使用它的serviceInterface);客户端然后可以调用代理上的方法,导致在AMQP上远程执行.

重要

使用默认 MessageConverter 器,方法参数和返回值必须是Serializable的实例.

在服务器端,AmqpInvokerServiceExporter 包含AmqpTemplate 和 MessageConverter属性. 

目前,未使用模板的MessageConverter.如果你需要提供定制的消息转换器,那么你需要使用messageConverter 属性进行提供.在客户端,可在AmqpTemplate 中添加定制消息转换器,它是使用其amqpTemplate 属性提供给 AmqpProxyFactoryBean 的.

样例 client 和server 配置如下所示.

 
   
 
   
 
   
 
   
重要
AmqpInvokerServiceExporter 只能处理适当格式的消息,如果从AmqpProxyFactoryBean中发出的消息. 如果它接收到一个不能解释的消息,那么将发送一个序列化的RuntimeException 作为回复.
如果这些消息无replyToAddress 属性,消息会被拒绝且在没有配置死信交换器时会永久丢失.
默认情况下,如果请求消息不能投递,调用线程最终会超时,并会抛出RemoteProxyFailureException. 超时时间是5秒,可在RabbitTemplate通过设置replyTimeout 属性来修改.
从1.5版本开始,如果设置 mandatory 属性为true, 并在连接工厂中启用了返回(参考 the section called “Publisher Confirms and Returns”), 调用线程会抛出一个AmqpMessageReturnedException. 
参考 the section called “Reply Timeout” 来了解更多信息.

 

-第六部分

3.1.10 配置broker

介绍

AMQP 规范描述了协议是如何用于broker中队列,交换器以及绑定上的.这些操作是从0.8规范中移植的,更高的存在于org.springframework.amqp.core包中的AmqpAdmin 接口中. 
那个接口的RabbitMQ 实现是RabbitAdmin,它位于org.springframework.amqp.rabbit.core 包.

AmqpAdmin接口是基于Spring AMQP 域抽象,展示如下:

public interface AmqpAdmin {      
// Exchange Operations
void declareExchange(Exchange exchange);      
void deleteExchange(String exchangeName);      
// Queue Operations      
Queue declareQueue();      
String declareQueue(Queue queue);      
void deleteQueue(String queueName);      
void deleteQueue(String queueName, boolean unused, boolean empty);      
void purgeQueue(String queueName, boolean noWait);      
// Binding Operations
void declareBinding(Binding binding);      
void removeBinding(Binding binding);      
Properties getQueueProperties(String queueName);  
}

getQueueProperties() 方法会返回关于队列的的一些有限信息(消息个数和消费者数目). 属性返回的keys像RabbitTemplate (QUEUE_NAMEQUEUE_MESSAGE_COUNTQUEUE_CONSUMER_COUNT)中的常量一样是可用的. 
RabbitMQ REST API 提供了更多关于 QueueInfo 对象的信息.

无参 declareQueue() 方法在broker上定义了一个队列,其名称是自动生成的. 自动生成队列的其它属性是exclusive=trueautoDelete=true, and durable=false.

declareQueue(Queue queue) 方法接受一个 Queue 对象,并且返回声明队列的名称.如果提供的队列名称是空字符串,broker 使用生成的名称来声明队列再将名称返回给调用者. Queue 对象本身是不会变化的. 

这种功能只能用于编程下直接调用RabbitAdmin. 它不支持在应用程序上下文中由admin来定义队列的自动声明.

与此形成鲜明对比的是,AnonymousQueue,框架会为其生成唯一名称(UUID),durable为false,exclusiveautoDelete 为true的匿名队列 带空的或缺失的name 属性总会创建 一个AnonymousQueue.

参考the section called “AnonymousQueue” 来理解为什么 AnonymousQueue 会优先选择broker生成队列名称,以及如何来控制名称格式. 声明队列必须有固定的名称,因为它们可能会上下文的其它地方引用,例如,在监听器中:

 
   

参考 the section called “Automatic Declaration of Exchanges, Queues and Bindings”.

此接口的RabbitMQ实现是RabbitAdmin,当用Spring XML配置时,看起来像下面这样:

CachingConnectionFactory 缓存模式是CHANNEL 时(默认的),  RabbitAdmin 实现会在同一个ApplicationContext中自动延迟声明 Queues,Exchanges 和 Bindings
只要Connection打开了与Broker的连接,这些组件就会被声明.有一些命名空间特性可以使这些变得便利,如,在Stocks 样例程序中有:

 
   
 
   
 
   
 
   

在上面的例子中,我们使用匿名队列(实际上由框架内部生成,而非由broker生成的队列),并用ID进行了指定.我们也可以使用明确的名称来声明队列,也作为上下文中bean定义的标识符.如.

重要
你可以提供id 和 name 属性.这允许你独立于队列名称通过id来指定队列.它允许使用标准的Spring 属性,如属性占位符和队列名称的SpEL 表达式; 当使用名称来作为标识符,这些特性是不可用的.

队列也可以使用其它的参数进行配置,例如x-message-ttl 或 x-ha-policy.通过命名空间支持,它们可以通过元素以参数名/参数值的MAP形式来提供 .

 
   
 
   

默认情况下,参数假设为字符串.对于其它类型的参数,需要提供类型.

 
   
 
   

当提供混合类型的参数时,可为每个entry元素提供type:

100
 
   
 
   
 
   

在Spring Framework 3.2或以后,声明起来更加简洁:

 
   
 
   
重要
RabbitMQ broker 不允许使用不匹配的参数来声明队列. 例如,如果一个无time to live参数的队列已经存在,然后你试图使用 key="x-message-ttl" value="100"进行声明,那么会抛出一个异常.

默认情况下,当出现异常时, RabbitAdmin 会立即停止所有声明的处理过程;这可能会导致下游问题- 如监听器容器会初始化失败,因另外的队列没有声明.

这种行为可以通过在RabbitAdmin上设置 ignore-declaration-exceptions 为true来修改. 此选项会指示RabbitAdmin 记录异常,并继续声明其它元素.当使用Java来配置RabbitAdmin 时, 此属性为ignoreDeclarationExceptions.
这是一个全局设置,它将应用到所有元素上,如应用到queues, exchanges 和bindings这些具有相似属性的元素上.

在1.6版本之前, 此属性只会在channel上发生IOExcepton时才会起作用- 如当目前和期望属性发生错配时. 现在, 这个属性可在任何异常上起作用,包括TimeoutException 等等.

此外,任何声明异常都会导致发布DeclarationExceptionEvent, 这是一个ApplicationEvent ,在上下文中可通过任何ApplicationListener 消费. 此事件包含了admin的引用, 正在声明的元素以及Throwable.

从1.3版本开始, HeadersExchange 可配置匹配多个headers; 你也可以指定是否需要必须匹配任何一个或全部headers:

 
   
 
   
 
   
 
   

从1.6版本开始,Exchanges 可使用internal 标志来配置(默认为false) ,当然,这样的Exchange 也可以通过 RabbitAdmin 来配置(如果在应用程序上下文中存在).
如果对于交换器来说,internal 标志为true , RabbitMQ 会允许客户端来使用交换器.这对于死信交换器来说或交换器到交换器绑定来说,是很用的,因为在这些地方你不想让发布者直接使用交换器.

要看如何使用Java来配置AMQP基础设施,可查看Stock样例程序,在那里有一个带@Configuration 注解的抽象AbstractStockRabbitConfiguration 类,它依次有RabbitClientConfiguration 和 RabbitServerConfiguration 子类. AbstractStockRabbitConfiguration 的代码展示如下:
 

@Configuration
public abstract class AbstractStockAppRabbitConfiguration {

    @Bean
 public ConnectionFactory connectionFactory() {
        CachingConnectionFactory connectionFactory =
            new CachingConnectionFactory("localhost");
        connectionFactory.setUsername("guest");
        connectionFactory.setPassword("guest");
        return connectionFactory;
    }

    @Bean
 public RabbitTemplate rabbitTemplate() {
        RabbitTemplate template = new RabbitTemplate(connectionFactory());
        template.setMessageConverter(jsonMessageConverter());
        configureRabbitTemplate(template);
        return template;
    }

    @Bean
 public MessageConverter jsonMessageConverter() {
        returnnew JsonMessageConverter();
    }

    @Bean
 public TopicExchange marketDataExchange() {
        returnnew TopicExchange("app.stock.marketdata");
    }

    // additional code omitted for brevity

}

在Stock 程序中,服务器使用下面的@Configuration注解来配置:

@Configuration
public class RabbitServerConfiguration extends AbstractStockAppRabbitConfiguration  {

    @Bean
 public Queue stockRequestQueue() {
        returnnew Queue("app.stock.request");
    }
}

这是整个@Configuration 类继承链结束的地方. 最终结果是TopicExchange 和队列会在应用程序启动时被声明.在服务器配置中,没有TopicExchange与队列的绑定,因为这是在客户端程序完成的.
然后stock 请求队列是自动绑定到AMQP 默认交换器上的 - 这种行为是由规范来定义的.

客户端 @Configuration 类令人关注的地方展示如下.
 

@Configuration
public class RabbitClientConfiguration extends AbstractStockAppRabbitConfiguration {

    @Value("${stocks.quote.pattern}")
 private String marketDataRoutingKey;

    @Bean
 public Queue marketDataQueue() {
        return amqpAdmin().declareQueue();
    }

    /**
     * Binds to the market data exchange.
     * Interested in any stock quotes
     * that match its routing key.
     */@Bean
  public Binding marketDataBinding() {
        return BindingBuilder.bind(
                marketDataQueue()).to(marketDataExchange()).with(marketDataRoutingKey);
    }

    // additional code omitted for brevity

}

客户端使用AmqpAdmin的declareQueue()方法声明了另一个队列,并将其绑定到了market data 交换器上(路由键模式是通常外部properties文件来定义的).

Queues 和Exchanges的Builder API

当使用Java配置时,1.6版本引入了一个便利的API来配置Queue 和Exchange 对象:

@Bean
public Queue queue() {
    return QueueBuilder.nonDurable("foo")
        .autoDelete()
        .exclusive()
        .withArgument("foo", "bar")
        .build();
}

@Bean
public Exchange exchange() {
  return ExchangeBuilder.directExchange("foo")
      .autoDelete()
      .internal()
      .withArgument("foo", "bar")
      .build();
}

查看 org.springframework.amqp.core.QueueBuilder 

和 org.springframework.amqp.core.ExchangeBuilder 的JavaDoc来了解更多信息.

Declaring Collections of Exchanges, Queues, Bindings

从1.5版本开始,可以在一个@Bean声明多个条目来返回集合.

只有集合中的第一个元素可认为是Declarablea的,并且只有集合中的Declarable 元素会被处理.(

Only collections where the first element is a Declarable are considered, and only Declarable elements from such collections are processed.)

@Configuration
public static class Config {

    @Bean
public ConnectionFactory cf() {
        returnnew CachingConnectionFactory("localhost");
    }

    @Bean
public RabbitAdmin admin(ConnectionFactory cf) {
        returnnew RabbitAdmin(cf);
    }

    @Bean
public DirectExchange e1() {
    	returnnew DirectExchange("e1", false, true);
    }

    @Bean
public Queue q1() {
    	returnnew Queue("q1", false, false, true);
    }

    @Bean
public Binding b1() {
    	return BindingBuilder.bind(q1()).to(e1()).with("k1");
    }

    @Bean
public List es() {
    	return Arrays.asList(
    			new DirectExchange("e2", false, true),
    			new DirectExchange("e3", false, true)
    	);
    }

    @Bean
public List qs() {
    	return Arrays.asList(
    			new Queue("q2", false, false, true),
    			new Queue("q3", false, false, true)
    	);
    }

    @Bean
public List bs() {
    	return Arrays.asList(
    			new Binding("q2", DestinationType.QUEUE, "e2", "k2", null),
    			new Binding("q3", DestinationType.QUEUE, "e3", "k3", null)
    	);
    }

    @Bean
public List ds() {
    	return Arrays.asList(
    			new DirectExchange("e4", false, true),
    			new Queue("q4", false, false, true),
    			new Binding("q4", DestinationType.QUEUE, "e4", "k4", null)
    	);
    }

}

条件声明

默认情况下,所有queues, exchanges,和bindings 都可通过应用程序上下文中所有RabbitAdmin 实例来声明(设置了auto-startup="true").

重要

从1.2版本开始,可以有条件地声明元素.当程序连接了多个brokers,并需要在哪些brokers上声明特定元素时,特别有用.

代表这些元素要实现Declarable 接口,此接口有两个方法: shouldDeclare() 和 getDeclaringAdmins()RabbitAdmin 使用这些方法来确定某个特定实例是否应该在其Connection上处理声明.

这些属性作为命名空间的属性是可用的,如下面的例子所示.

 
   
 
   
重要
默认情况下,如果没有提供declared-by(或是空的), auto-declare 属性则为 true,那么所有RabbitAdmin将声明对象(只要admin的auto-startup 属性为true,默认值).

现样的,你可以使用基于Java的@Configuration 注解来达到同样的效果.在这个例子中,组件会由admin1来声明,而不是admin2:

@Bean
public RabbitAdmin admin() {
	RabbitAdmin rabbitAdmin = new RabbitAdmin(cf1());
	rabbitAdmin.afterPropertiesSet();
	return rabbitAdmin;
}

@Bean
public RabbitAdmin admin2() {
	RabbitAdmin rabbitAdmin = new RabbitAdmin(cf2());
	rabbitAdmin.afterPropertiesSet();
	return rabbitAdmin;
}

@Bean
public Queue queue() {
	Queue queue = new Queue("foo");
	queue.setAdminsThatShouldDeclare(admin());
	return queue;
}

@Bean
public Exchange exchange() {
	DirectExchange exchange = new DirectExchange("bar");
	exchange.setAdminsThatShouldDeclare(admin());
	return exchange;
}

@Bean
public Binding binding() {
	Binding binding = new Binding("foo", DestinationType.QUEUE, exchange().getName(), "foo", null);
	binding.setAdminsThatShouldDeclare(admin());
	return binding;
}

AnonymousQueue

一般来说,当需要一个独特命名,专用的,自动删除队列时,建议使用AnonymousQueue 来代替中间件定义的队列名称(使用 "" 作为队列名称会导致中间件生成队列名称).

这是因为:

  1. 队列实际上是在与broker的连接建立时声明的;这在bean创建和包装之后要很长时间;使用这个队列的beans需要知道其名称.而事实上,当app启动时,broker甚至还没有运行.
  2. 如果与broker的连接因某种原因丢失了,admin会使用相同的名称会重新声明AnonymousQueue.如果我们使用broker-声明队列,队列名称可能会改变.

从1.5.3版本开始,你可通过AnonymousQueue 来控制队列名称的格式.

默认情况下,队列名称是UUID的字符串表示; 例如: 07afcfe9-fe77-4983-8645-0061ec61a47a.

现在,你可以提供一个 AnonymousQueue.NamingStrategy 实现作为其构造器参数:

@Bean
public Queue anon1() {
    return new AnonymousQueue(new AnonymousQueue.Base64UrlNamingStrategy());
}

@Bean
public Queue anon2() {
    return new AnonymousQueue(new AnonymousQueue.Base64UrlNamingStrategy("foo-"));
}

第一个会生成队列名称前辍spring.gen- 其后为UUID base64 的表示,例如:spring.gen-MRBv9sqISkuCiPfOYfpo4g. 第二个会生成队列名称前辍为foo- 其后为UUID的 base64 表示.

base64 编码使用RFC 4648的"URL and Filename Safe Alphabet" ; 删除了字符(=).

你可以提供你自己的命名策略, 可以包括队列名称中的其他信息(例如应用程序、客户端主机)。

从1.6版本开始,当使用XML配置时,可以指定命名策略; naming-strategy 属性出现在元素的属性中,对于bean引用来说,它们实现了AnonymousQueue.NamingStrategy.

 
   
第一个创建了UUID字符串表示的名称.第二个创建了类似spring.gen-MRBv9sqISkuCiPfOYfpo4g的名称. 第三个创建了类似custom.gen-MRBv9sqISkuCiPfOYfpo4g的名称.

当然,你可以提供你自己的命名策略bean.

3.1.11 延迟的消息交换器

1.6版本引入了 Delayed Message Exchange Plugin支持.

该插件目前被标记为实验性质,但可用已超过一年(在写作的时间)。如果插件的变化是必要的,我们将尽快添加支持这样的变化。因此,这种在Spring AMQP支持同样也应考虑为实验性质.这个功能在RabbitMQ 3.6.0版本和0.0.1插件版本中经过测试。

要使用RabbitAdmin 来声明一个延迟交换器,只需要在交换器上简单地设置delayed 属性为true. RabbitAdmin 会使用交换器类型(DirectFanout 等)来设置x-delayed-type 参数,并使用x-delayed-message来声明交换器.

当使用XML来配置交换器beans时,delayed 属性 (默认为false)是可用的.

要发送延迟消息,只需要通过MessageProperties设置x-delay header:

MessageProperties properties = new MessageProperties();
properties.setXDelay(15000);
template.send(exchange, routingKey,
        MessageBuilder.withBody("foo".getBytes()).andProperties(properties).build());

rabbitTemplate.convertAndSend(exchange, routingKey, "foo", new MessagePostProcessor() {

    @Override
public Message postProcessMessage(Message message) throws AmqpException {
        message.getMessageProperties().setXDelay(15000);
        return message;
    }

});

要检查消息是否是延迟的,可调用MessageProperties的getReceivedDelay(). 它是一个独立的属性,以避免从一个输入消息意外的传播到一个输出消息。
 

3.1.12 RabbitMQ REST API

当启用了管理插件时,RabbitMQ 服务器公开了 REST API 来监控和配置broker. 

现在提供了 Java Binding for the API.一般来说,你可以直接使用API,但提供了便利的包装器来使用熟悉的Spring AMQP QueueExchange, 和 Binding 域对象.
当直接使用 com.rabbitmq.http.client.Client API  (分别使用QueueInfoExchangeInfo, 和BindingInfo),那些对象的更多信息将可用. 下面是RabbitManagementTemplate上的可用操作:

public interface AmqpManagementOperations {

	void addExchange(Exchange exchange);

	void addExchange(String vhost, Exchange exchange);

	void purgeQueue(Queue queue);

	void purgeQueue(String vhost, Queue queue);

	void deleteQueue(Queue queue);

	void deleteQueue(String vhost, Queue queue);

	Queue getQueue(String name);

	Queue getQueue(String vhost, String name);

	List getQueues();

	List getQueues(String vhost);

	void addQueue(Queue queue);

	void addQueue(String vhost, Queue queue);

	void deleteExchange(Exchange exchange);

	void deleteExchange(String vhost, Exchange exchange);

	Exchange getExchange(String name);

	Exchange getExchange(String vhost, String name);

	List getExchanges();

	List getExchanges(String vhost);

	List getBindings();

	List getBindings(String vhost);

	List getBindingsForExchange(String vhost, String exchange);

}

参考javadocs 来了解更多信息.

3.1.13 异常处理

RabbitMQ Java client的许多操作会抛出受查异常. 例如,有许多可能抛出IOExceptions的地方. RabbitTemplate, SimpleMessageListenerContainer, 和其它Spring AMQP 组件会捕获这些异常,并将它们转换为运行时层次的异常. 
这些是定义在org.springframework.amqp 包中的,且 AmqpException 是层次结构的基础.

当监听器抛出异常时,它会包装在一个 ListenerExecutionFailedException 中,正常情况下消息会被拒绝并由broker重新排队.将defaultRequeueRejected 设置为false 可导致消息丢弃(或路由到死信交换器中). 

正如 the section called “Message Listeners and the Asynchronous Case”讨论的,监听器可抛出 AmqpRejectAndDontRequeueException 来有条件地控制这种行为。

然而,有一种类型的错误,监听器无法控制其行为. 当遇到消息不能转换时(例如,无效的content_encoding 头),那么消息在到达用户代码前会抛出一些异常.当设置 defaultRequeueRejected 为 true (默认),这样的消息可能会一遍又一遍地重新投递.
在1.3.2版本之前,用户需要编写定制ErrorHandler, 正如Section 3.1.13, “Exception Handling” 描述的内容来避免这种情况.

从1.3.2版本开始,默认的ErrorHandler 是 ConditionalRejectingErrorHandler ,它将拒绝那些失败且不可恢复的消息 (不会重新排队):

  • o.s.amqp...MessageConversionException
  • o.s.messaging...MessageConversionException
  • o.s.messaging...MethodArgumentNotValidException
  • o.s.messaging...MethodArgumentTypeMismatchException

第一个是在使用MessageConverter转换传入消息负荷时抛出的.
第二个是当映射到@RabbitListener方法时,转换服务需要其它转换抛出的.
第三个是在监听器上使用了验证(如.@Valid),且验证失败时抛出的. 
第四个是对于目标方法传入消息类型转换失败抛出的.例如,参数声明为Message ,但收到的是Message

错误处理器的实例可使用FatalExceptionStrategy 来配置,因为用户可以提供它们的规则来有条件的拒绝消息,如. 来自 Spring Retry (the section called “Message Listeners and the Asynchronous Case”)中的BinaryExceptionClassifier代理实现. 
此外, ListenerExecutionFailedException 现在有一个可用于决策的failedMessage 属性.如果FatalExceptionStrategy.isFatal() 方法返回true,错误处理器会抛出AmqpRejectAndDontRequeueException.
默认FatalExceptionStrategy 会记录warning信息.

3.1.14 事务(Transactions)

介绍

Spring Rabbit 框架支持在同步和异步使用中使用不同语义(这一点对于现有Spring事务的用户是很熟悉的)来支持自动事务管理. 它做了很多,不是常见消息模式能轻易实现的.

有两种方法可用来向框架发出期望事务语义的信号.在RabbitTemplate 和 SimpleMessageListenerContainer 中,这里有一个channelTransacted 标记,如果它为true,就会告知框架使用事务通道,并根据结果使用提交或回滚来结束所有操作,出现异常时则发出回滚信号. 

另一个提供的信号是Spring的PlatformTransactionManager实现(作为正在进行的操作的上下文)的外部事务. 
当框架发送或接收消息时,如果过程中已经存在一个事务,且channelTransacted 标记为true, 那么当前消息事务的提交或回滚操作会延迟直到在当前事务结束.如果channelTransacted 标记为false,那么消息操作是不会应用事务语义(它是自动应答的).

channelTransacted 标记是一个配置时设置:它只在AMQP组件声明时执行一次,通常在应用程序启动时.原则上,外部事务更加动态化,因为需要在运行时根据当前线程状态来响应,当事务分层到应用程序上时,原则上来说它通常也是一个配置设置.

对于使用RabbitTemplate 的同步使用,外部事务是由调用者提供的, 要么是声明的,要么是强制的(日常Spring事务模式). 

下面是声明方法的一个例子(通常选择这个,因为它是非侵入的), 下面的例子中,模板已经配置了channelTransacted=true:

@Transactional
public void doSomething() {
    String incoming = rabbitTemplate.receiveAndConvert();
    // do some more database processing...
    String outgoing = processInDatabaseAndExtractReply(incoming);
    rabbitTemplate.convertAndSend(outgoing);
}

收到字符负荷,转换,并以消息体发送到@Transactional标记的方法中,因此如果数据处理因异常失败了,传入消息将返回到broker,并且输出消息不会被发送. 
在事务方法链中,这适用于RabbitTemplate 中的所有操作(除非Channel 较早地直接控制了提交事务).

对于SimpleMessageListenerContainer 的异步使用情况,如果需要外部事务,当设置了监听器时,必须由容器来发出请求. 
为了表示需要外部事务,当配置时,用户为容器提供了PlatformTransactionManager 实现.例如:

@Configuration
public class ExampleExternalTransactionAmqpConfiguration {

@Bean
public SimpleMessageListenerContainer messageListenerContainer() {
        SimpleMessageListenerContainer container = new SimpleMessageListenerContainer();
        container.setConnectionFactory(rabbitConnectionFactory());
        container.setTransactionManager(transactionManager());
        container.setChannelTransacted(true);
        container.setQueueName("some.queue");
        container.setMessageListener(exampleListener());
        return container;
    }

}

在上面的例子中,事务管理器是通过其它bean中注入添加的(未显示),并且channelTransacted 也设置为了true.其效果是如果监听器因异常失败了,那么事务将回滚,消息也会退回到broker中. 
明显地,如果事务提交失败(如.数据库约束错误,或通过问题),那么AMQP 事务也要回滚,且消息也会回退到broker中. 
有时候,这被称为最好努力1阶段提交(Best Efforts 1 Phase Commit),它是可靠消息非常强大的模式.
如果在上面的例子中将channelTransacted标志设为false(默认为false),那么外部事务仍会提供给监听器,但所有消息操作都是自动应答的, 因此其效果是即使发生了业务操作,也会提供消息操作.

 

关于接收消息的回滚说明

AMQP 事务只适用于发送应答给broker, 所以当有 Spring 事务回滚且又收到了消息时,Spring AMQP做的不仅要回滚事务,还要手动拒绝消息. 
消息上的拒绝操作独立于事务,依赖于defaultRequeueRejected 属性(默认为true). 更多关于拒绝失败消息的详情,请参考the section called “Message Listeners and the Asynchronous Case”.

关于RabbitMQ 事务及其局限性的更多信息,参考RabbitMQ Broker Semantics.

重要

在 RabbitMQ 2.7.0前, 这样的消息(当通道关闭或中断时未应的消息)会回到队列中,从2.7.0, 拒绝消息会跑到队列前边,与JMS回滚消息方式类似.

使用RabbitTransactionManager

RabbitTransactionManager 是执行同步,外部事务Rabbit操作的另一种选择.这个事务管理器是PlatformTransactionManager 接口的实现类,应该在单个Rabbit ConnectionFactory中使用.

重要

此策略不能提供XA事务,比如,要在消息和数据库之间共享事务.

应用代码需要通过ConnectionFactoryUtils.getTransactionalResourceHolder(ConnectionFactory, boolean)来获取事务性Rabbit资源而不是使用Connection.createChannel() 调用.
当使用Spring AMQP的 RabbitTemplate时, 它会自动检测线程绑定通道和自动参与事务。

在 Java 配置中,你可以使用下面的代码来设置一个新的RabbitTransactionManager:

@Bean
public RabbitTransactionManager rabbitTransactionManager() {
    returnnew RabbitTransactionManager(connectionFactory);
}

如果你喜欢使用XML 配置,可以像下面进行声明:

 
    
   

3.1.15 消息监听器容器配置

有相当多的配置SimpleMessageListenerContainer 相关事务和服务质量的选项,它们之间可以互相交互.当使用命名空间来配置时,

下表显示了容器属性名称和它们等价的属性名称(在括号中).

未被命名空间暴露的属性,以`N/A`表示.

 

-第七部分

3.1.15 消息监听器容器配置

有相当多的配置SimpleMessageListenerContainer 相关事务和服务质量的选项,它们之间可以互相交互.当使用命名空间来配置时,

下表显示了容器属性名称和它们等价的属性名称(在括号中).

未被命名空间暴露的属性,以`N/A`表示.
Table 3.3. 消息监听器容器的配置选项
属性
(group)
描述

 

只在使用命名空间时可用. 当经过指定时,类型为Collection 的bean会使用它这个名称进行注册,容器会将每个 元素添加到集合中.这就允许,如,通过迭代集合来启动/停止该组容器.如果多个 元素有相同的group值, 那么集合中的容器是所有指定容器的总和.

属性

 

 

channelTransacted(channel-transacted)

 

描述

Boolean标志,用于表示在事务中的所有消息都应该应答(手动地或自动地)

 

 

属性

 

 
acknowledgeMode(acknowledge)

 

描述

 

  • NONE: 不发送应答(与channelTransacted=true不兼容). RabbitMQ 称此为 "autoack",因为broker假设消费者没有采取任何动作应答了所有消息.
  • MANUAL:监听器必须调用Channel.basicAck()来手动应答所有消息.
  • AUTO :容器会自动应答所有消息, 除非MessageListener 抛出了异常. 注意acknowledgeMode 与channelTransacted 是互补的- 如果通道是事务的,那么broker除了ack外,还需要提交通知. 这是默认模式. 也可参考txSize.

 

属性

 

transactionManager(transaction-manager)

 

描述

监听器操作的外部事务管理器. 也是与channelTransacted互补的 -如果通道是事务的,那么其事务会用外部事务来同步.

 

 

属性

 

prefetchCount(prefetch)

 

描述

 

可接受来自broker一个socket帧中的消息数目. 数值越大,消息分发速度就越快, 但无序处理的风险更高. 

如果acknowledgeMode为NONE它会忽略. 它会增长,如果有必要,须匹配txSize.

 

 

 

 

属性

 

shutdownTimeout(N/A)

 

描述
当容器关闭时(例如. 关闭ApplicationContext),用于等待正在传输消息的上限时间.默认是5秒. 当达到上限时,如果通道是非事务的,消息将被丢弃.

 

 

属性

 

txSize(transaction-size)

 

描述
acknowledgeMode 为AUTO时,在发送ack前(等待每一个消息达到接收超时设置),容器将试图处理这个数目的消息 . 当事务通道提交后也是一样的.如果prefetchCount 小于txSize,prefetchCount 会增长以匹配txSize.

属性

receiveTimeout(receive-timeout)

描述

等待消息的最大时间.如果acknowledgeMode=NONE 这只有很小的效果 - 容器只旋转一轮,并要求另一个消息. 当在txSize>1的事务通道中有最大效果,因为它能导致已经消费但没有应答的消息直接超时过期.

 

 

属性

 

autoStartup(auto-startup)

 

描述

用于当ApplicationContext启动时(作为SmartLifecycle 回调的一部分,发生在所有bean初始化之后)是否同时启动容器的标志.默认为true,如果在容器启动时,中间件暂不可用,那么可将其设为false,随后在确认中间件已启动后,手动调用start() 方法来启动.

 

 

属性

phase(phase)

 

 

描述

 

当autoStartup为true时,容器中的生命周期阶段应该启动和停止.值越小,容器就会越早启动,并越晚停止.默认值Integer.MAX_VALUE,这意味着容器会越晚启动并尽快停止.

 

属性

adviceChain(advice-chain)

描述

 

应用于监听器执行路径上的AOP Advice数组. 它可用于额外的横切关注点,如broker死亡事件中的自动重试. 

注意,只要broker还活着,出现AMQP错误后的重新连接是由CachingConnectionFactory来处理的.

 

属性

 

taskExecutor(task-executor)

 

描述

执行监听器调用程序的Spring TaskExecutor引用(或标准JDK 1.5+ Executor). 默认是 SimpleAsyncTaskExecutor, 用于内部管理线程.

 

 

属性

 

errorHandler(error-handler)

 

描述

在MessageListener执行期间,用于处理未捕获异常的ErrorHandler策略的引用. 默认是 ConditionalRejectingErrorHandler.

 

 

属性

 

concurrentConsumers(concurrency)

 

描述
每个监听器上初始启动的并发消费者数目. 参考Section 3.1.16, “Listener Concurrency”.

 

 

属性

 

axConcurrentConsumers(max-concurrency)

 

描述

启动并发消费者的最大数目,如果有必要,可以按需设置.必须要大于或等于concurrentConsumers

参考Section 3.1.16, “Listener Concurrency”.

属性

 

startConsumerMinInterval(min-start-interval)

 

描述
启动新消费者之间的时间间隔,单位为毫秒. 
参考 Section 3.1.16, “Listener Concurrency”. 默认 10000 (10 秒).

属性

 

stopConsumerMinInterval(min-stop-interval)

 

描述

停止消费者的时间间隔, 由于最后一个消费者已经停止了,这时可以检测到空闲消费者.

参考Section 3.1.16, “Listener Concurrency”. 默认 60000 (1 分钟).

属性

 

consecutiveActiveTrigger(min-consecutive-active)

 

描述

消费者收到连续消息的最小数量,当考虑启动一个新的消费者,接收不会发生超时。也会受txsize影响. 参考 Section 3.1.16, “Listener Concurrency”. 默认为10.

属性

 

consecutiveIdleTrigger(min-consecutive-idle)

 

描述

在考虑停止一个消费者,消费者必须经历的最小接收超时时间,也会受txsize影响. 

参考 Section 3.1.16, “Listener Concurrency”. 默认为10.

属性

 

connectionFactory(connection-factory)

 

描述
connectionFactory的引用; 当使用XML命名空间配置时,默认引用bean名称是"rabbitConnectionFactory".

属性

 

defaultRequeueRejected(requeue-rejected)

 

描述

用以确定因监听器抛出异常而遭受拒绝的消息是否需要重新入列. 默认为true.

属性

recoveryInterval(recovery-interval)

描述

如果消费者不是因致命原因而导致启动失败,则用于设置重启消费者的时间间隔,单位毫秒. 默认为5000.与recoveryBackOff互斥.

属性

recoveryBackOff(recovery-back-off)

描述

如果消费者不是因致命原因而导致启动失败,则用于指定 BackOff 启动消费者的时间间隔. 默认是每5秒无限重试的FixedBackOff. 与recoveryInterval互斥.

属性

exclusive(exclusive)

描述

用于确定容器中的单个消费者是否具有独占访问队列的权限。当其值为1时,容器的concurrency必须为1时。如果另一个消费者有独占访问权,容器将根据恢复时间间隔或恢复后退试图恢复消费者。
当使用命名空间时,此属性会随着队列名称出现在元素中。默认为false。

属性

rabbitAdmin(admin)

描述

监听器监听了多个自动删除队列时,当其发现在启动时队列消失了,容器会使用RabbitAdmin 来声明消失的队列,并进行交换器的相关绑定. 
如果此元素配置成使用条件声明(参考the section called “Conditional Declaration”), 容器必须使用配置的admin来声明那些元素. 
这里指定的admin;只在使用带有条件声明的自动删除队列时才需要. 如果你不想在容器启动前声明自动删除队列,可在amdin中将 auto-startup 设为false. 默认情况下,RabbitAdmin 会声明所有非条件元素.

属性

missingQueuesFatal(missing-queues-fatal)

描述

从1.3.5版本开始,SimpleMessageListenerContainer 就有了这个新属性.

当设为true (默认值)时,如果配置队列在中间件都不可用, 这会视为是致命的.这会导致应用程序上下文初始化失败; 同时, 当容器还在运行时删除了队列,也会发生这样的情况.
默认情况下,消费者进行3次重试来连接队列(5秒时间间隔),如果所有尝试都失败了则会停止容器.

在以前版本中,此选项是不可配置的.

当设置为false, 再做了三次重试后,容器将进入恢复模式, 这也伴随其它问题,如中间件已经发生了故障.

容器会根据recoveryInterval 属性来尝试恢复. 在每次恢复尝试期间,每个消费者会以5秒的时间间隔来尝试4次被动声明. 这个过程将无限期地继续下去(译者注:有点冲突)。

你也可以使用properties bean来为所有的容器全局设置属性,如下所示:

false
 
   

如果容器明确的设置了 missingQueuesFatal 属性,全局属性的值对此容器将无效.

默认的retry属性(5秒间隔3次重试)可通过下面的属性值来覆盖.

属性

mismatchedQueuesFatal(mismatched-queues-fatal)

描述

这是1.6版本中加入的新属性.当容器启动时,如果此属性为true (默认为false), 容器会检查上下文中声明的队列是否中间件中存在的队列是否一致.

如果属性不匹配(如. auto-delete) 或参数 (e.g. x-message-ttl) 存在, 容器 (和应用程序上下文) 会抛出致命异常而导致启动失败.如果是在恢复期间检测到的问题,容器会停止.

必须在上下文中存在单个RabbitAdmin (或使用rabbitAdmin属性在容器上特别配置);否则此属性必须为false.

如果在初始启动期间,中间件还不可用,容器启动后,当建立连接时会检查条件.

重要

该检查针对的是上下文的所有队列,而不仅仅是特定监听器配置使用的队列.如果你希望只检查容器使用的队列,你需要为这个容器配置单独的RabbitAdmin , 并使用rabbitAdmin 属性为其提供一个引用. 

参考“Conditional Declaration”章节来了解更多信息.

属性

 

autoDeclare(auto-declare)

描述

 

 

从1.4版本开始, SimpleMessageListenerContainer 引入了这个新属性.

当设置为true 时(默认值),容器会使用RabbitAdmin 来重新声明所有 AMQP 对象(Queues, Exchanges, Bindings).
如果在启动期间探测到至少有一个队列缺失了,可能因为它是自动删除队列或过期队列,但不管队列缺失是基于什么原因,重新声明仍会进行处理(译者注:太浪费了).
要禁用这种行为, 可设置其属性为false. 但需要注意的是,如果所有队列都缺失了(译者注:全部还是部分),容器会启动失败.

在1.6版本之前,如果在上下文中存在多个admin,容器会随机选择一个.反之,如果没有admin,它会从内部创建一个.
无论是哪种情况,这都将导致非预期结果出现. 从1.6版本开始,为了能使autoDeclare 工作,必须要上下文中明确存在一个RabbitAdmin,或者特定实例的引用必须要在容器中使用rabbitAdmin属性中配置.

 

属性

 

declarationRetries(declaration-retries)

描述

 

 

从1.4.3, 1.3.9版本开始,SimpleMessageListenerContainer 有了这个新属性. 命名空间属性在1.5.x中可用.

用于设置被动声明失败时,重新尝试的次数.被动声明发生在当消费者启动了或从多个队列中消费时,初始化期间部分队列还不可用的情况下. 
当重试次数用完后,如果还是不能被动声明配置队列,那么上面的missingQueuesFatal属性将控制容器行为. 默认: 3次重试 (4 次尝试).

属性

 

 

failedDeclarationRetryInterval(failed-declaration-retry-interval)

描述

 

 

从1.4.3, 1.3.9版本开始,SimpleMessageListenerContainer 有了这个新属性. 命名空间属性在1.5.x中可用.

重新尝试被动声明的时间间隔. 被动声明发生在当消费者启动了或从多个队列中消费时,初始化期间部分队列还不可用的情况下. 默认: 5000 (5秒).

属性

 

 

retryDeclarationInterval(missing-queue-retry-interval)
描述

从1.4.3, 1.3.9版本开始,SimpleMessageListenerContainer 有了这个新属性. 命名空间属性在1.5.x中可用.

 

 

如果配置队列的一个子集在消费者初始化过程中可用,则消费者将从这些队列中开始消费。消费者将被动地使用此间隔声明丢失的队列。

当这个间隔过去后,会再次使用declarationRetries 和 failedDeclarationRetryInterval

如果还有缺失队列,消费者在重新尝试之前会等待此时间间隔. 

这个过程会不停地进行到所有队列可用. 默认: 60000 (1分钟).

属性

 

 

consumerTagStrategy(consumer-tag-strategy)

描述

 

 

从1.4.5版本开始,SimpleMessageListenerContainer 有了这个新属性. 命名空间属性在1.5.x中可用.

之间,只能使用中间件生成的consumer tags;尽管现在这仍是默认的配置,但现在你可以提供一个ConsumerTagStrategy的实现, 这样就可为每个消费者创建独特的tag.

属性

 

 

idleEventInterval(idle-event-integer)

描述

 

从1.6版本开始,SimpleMessageListenerContainer 有了这个新属性. 

参考"Detecting Idle Asynchronous Consumers"章节.
 

3.1.16 监听器并发

默认情况下,监听器容器会启动单个消费者来接收队列中的消息.

当检查前面章节中的表格时,你会发现有许多属性可控制并发.最简单的是concurrentConsumers, 它会创建固定数量的消费者来并发处理消息.

在1.3.0版本之前,这只能在容器停止时才可设置.

从1.3.0版本开始,你可以动态调整 concurrentConsumers 属性.如果容器运行时修改了,会根据新设置来调需要的消费者(添加或删除).

此外,在容器中添加了一个新属性 maxConcurrentConsumers 来基于工作负载来动态调整并发数. 

它可与其它四个属性一起工作: consecutiveActiveTriggerstartConsumerMinIntervalconsecutiveIdleTriggerstopConsumerMinInterval
在默认设置的情况下,加大消费者的算法如下:

如果还没有达到maxConcurrentConsumers ,如果现有消费者活动了10个连续周期且离最后消费者启动至少消逝了10秒钟,那么将启动新的消费者. 如果消费者在txSize * receiveTimeout 毫秒内至少收到一个消息,那么就认为此消费者是活动的.

在默认设置的情况下,减少消费者的算法如下:

如果有超过concurrentConsumers 数量的消费者在运行,且检测到消费者连续超时(空闲)了10个周期,且最后一个消费者至少停止了60秒,那么消费者将停止.
超时依赖于receiveTimeout 和 txSize 属性.当在txSize * receiveTimeout 毫秒内未收到消息,则认为消费者是空闲的.
因此,当有默认超时(1秒)和 txSize为4,那么在空闲40秒后,会认为消费者是空闲的并会停止(4超时对应1个空闲检测).

 

实际上,如果整个容器空闲一段时间,消费者将只会被停止。这是因为broker将分享其在所有活跃的消费者的工作。

3.1.17 专用消费者

也是从1.3版本开始,监听器容器可配置单个专用消费者; 这可以阻其它容器来消费队列直到当前消费者退出. 

这样的容器的并发性必须是1。

当使用专用消费者时,其它容器会根据recoveryInterval 属性来消费队列, 如果尝试失败,会记录一个 WARNing 信息.

3.1.18 监听器容器队列

1.3版本在监听器容器中引入许多处理多个队列的改善措施.

容器配置必须监听至少一个队列以上; 以前也是这样的情况,但现在可以在运行时添加和删除队列了。当任何预先获取的消息被处理后,容器将回收(取消和重新创建)。
参考方法addQueuesaddQueueNamesremoveQueuesand removeQueueNames.当删除队列时,至少要保留一个队列.

现在,只要有可用队列消费者就会启动 -先前如果没有可用队列,容器会停止.现在,唯一的问题是是否有可用队列.如果只是部分队列可用,容器会每60秒尝试被动声明(和消费)缺失队列.

此外,如果消费才从broker中收到了通道(例如,队列被删除)消费者会尝试重新恢复,重新恢复的消费会继续处理来自其它配置队列中的消息. 之前是队列上的取消会取消整个消费者,最终容器会因缺失队列而停止.

如果你想永久删除队列,你应该在删除队列的之前或之后更新容器,以避免消费.

3.1.19 恢复:从错误和代理失败中恢复

介绍

Spring提供了一些关键的 (最流行的)高级特性来处理协议错误或中间件失败时的恢复与自动重连接. 

主要的重连接特性可通过CachingConnectionFactory 自身来开启. 它也常有利于使用rabbitadmin自动声明的特点. 
除此之外, 如果你关心保证投递,你也许需要在RabbitTemplate中使用channelTransacted 标记以及在SimpleMessageListenerContainer中使用AcknowledgeMode.AUTO (或者自己来手动应答) .

 

交换器、队列和绑定的自动声明

RabbitAdmin 组件在启动时可声明交换器,队列,绑定.它是通过ConnectionListener懒执行的,因此如果启动时broker不存在,也没有关系. 
Connection 第一次使用时(如.发送消息) ,监听器会被触发,admin功能也会应用.这种在监听器中自动声明的好处是,如果连接出于任何原因断开了,(如. broker死了,网络中断问题.),它们会在下次有需要的时候重新应用.

这种方式的队列声明必须要有固定的名称;要么是明确声明,要么是由框架生成AnonymousQueue.匿名队列是非持久化的,专用的,且自动删除的.

重要

自动声明只在cachingConnectionFactory 缓存模式是CHANNEL (默认)才可用. 这种限制的存在是因为专用和自动删除队列是绑定到connection上的.

 

同步操作中的故障和重试选项

如果你在同步序列中使用RabbitTemplate时丢失了broker的连接,那么Spring AMQP会抛出一个AmqpException (通常但并不总是AmqpIOException). 
我们不想隐藏存在问题的事实,因此你可以捕获并对异常进行处理.如果你怀疑连接丢失了,而且这不是你的错,那么最简单的事情就是执行再次尝试操作. 重试操作可以手动进行,也可以使用Spring Retry来处理重试(强制或声明).

Spring Retry 提供了两个AOP拦截器并提供非常灵活的方式来指定retry的参数(尝试的次数,异常类型, 补偿算法等等.). Spring AMQP同时也提供了一些方便的工厂bean来创建Spring Retry拦截器, 你可以使用强类型回调接口来实现恢复逻辑.参考Javadocs和 StatefulRetryOperationsInterceptor 和StatelessRetryOperationsInterceptor 的属性来了解更多详情. 

如果没有事务,或者如果一个事务是在重试回调中启动的话,则无状态重试是适当的。注意,相对于有状态重试,无状态重试只是简单配置和分析,如果存在一个正在进行的事务必须回滚或肯定会回滚的话, 这种无状态重试则是不合适的. 
在事务中间掉下来的连接与回退有同样的效果, 所以对于事务开始于堆栈上的重连接来说,有状态重试通常是最佳选择(so for reconnection where the transaction is started higher up the stack, stateful retry is usually the best choice).

从1.3版本开始,提供了builder API来帮助在Java中使用这些拦截器(或者在 @Configuration 类中),例如:

@Bean
public StatefulRetryOperationsInterceptor interceptor() {
	return RetryInterceptorBuilder.stateful()
			.maxAttempts(5)
			.backOffOptions(1000, 2.0, 10000) // initialInterval, multiplier, maxInterval
			.build();
}

只有部分retry特性能通过这种方式,更加高级的特性需要在RetryTemplate 中配置. 

参考Spring Retry Javadocs 来了解可用策略,配置的完整信息.

消息监听器和异步情况

如果 MessageListener 因业务异常而失败,异常可由消息监听器容器来处理,然后它会继续回去监听其它信息.如果失败是由于掉下的连接引起的(非业务异常),那么监听此消费者的监听器将退出和重启.

SimpleMessageListenerContainer 可以无逢地进行处理,并且它会在日志中记录监听器即将重启. 

事实上,它会循环不断地尝试重新启动消费者,只有当消费者有非常糟糕的行为时,才会放弃。一个副作用是,如果broker在容器启动时关闭,它将会继续尝试直到建立一个连接。

业务异常处理, 相对于协议错误和连接丢失,它可能需要更多考虑和一些自定义配置,特别是处于事务或 容器应答时.
在2.8.x版本之前, RabbitMQ对于死信行为没有定义,因此默认情况下,一个因拒绝或因业务异常导致回退的消息可循环往复地重新分发.
要限制客户端的重新分发的次数,一个选择是在监听器的通知链中添加一个StatefulRetryOperationsInterceptor. 拦截器有一个实现了自定义死信动作的恢复回调: 

什么是适合你的特定的环境。

另一个选择是设置容器的rejectRequeued属性为false. 这会导致丢弃所有失败的消息.当使用RabbitMQ 2.8.x+时,这也有利于传递消息到一个死的信件交换。

或者,你可以抛出一个AmqpRejectAndDontRequeueException;这会阻止消息重新入列,不管defaultRequeueRejected 属性设置的是什么.

通常情况下,可以组合使用这两种技术 在通知链中使用StatefulRetryOperationsInterceptor, 在此处是MessageRecover 抛出AmqpRejectAndDontRequeueException. MessageRecover 会一直调用,直到耗尽了所有重试. 
默认MessageRecoverer 只是简单的消费错误消息,并发出WARN消息.在这种情况下,消息是通过应答的,且不会发送到死信交换器中.

从1.3版本开始,提供了一个新的RepublishMessageRecoverer,它允许在重试次数耗尽后,发布失败消息:

@Bean
RetryOperationsInterceptor interceptor() {
	return RetryInterceptorBuilder.stateless()
			.maxAttempts(5)
			.recoverer(new RepublishMessageRecoverer(amqpTemplate(), "bar", "baz"))
			.build();
}

RepublishMessageRecoverer 会使用消息头的额外信息来发布,这些信息包括异常信息,栈轨迹,原始交换器和路由键.额外的头可通过创建其子类和覆盖additionalHeaders()方法来添加.

 

重试的异常分类

Spring Retry 可以非常灵活地决定哪些异常可调用重试. 默认配置是对所有异常都进行重试.用户异常可以包装在ListenerExecutionFailedException 中,我们需要确保分类检查异常原因. 默认的分类只是看顶部级别的异常。

从 Spring Retry 1.0.3开始BinaryExceptionClassifier 有一个属性traverseCauses (默认为false). 当当为true时,它将遍历异常的原因,直到它找到一个匹配或没有原因。

要使用分类重试,需要使用一个SimpleRetryPolicy ,其构造函数将接受最大尝试次数,Exception的Map,以及一个boolean值(traverseCauses),且还需要将此策略注入给RetryTemplate.

3.1.20 调试

Spring AMQP 提供广泛的日志记录,尤其是在DEBUG级别.

如果你想在应用程序和broker之间监控AMQP协议,你可以使用像WireShark的工具, 它有一个插件可用于解码协议.
另一个选择是, RabbitMQ java client自身携带了一个非常有用的工具类:Tracer.当以main方式运行时,默认情况下,它监听于5673 ,并连接本地的5672端口.
只需要简单的运行它,并修改你的连接工厂配置,将其连接到本地的5673端口. 它就会在控制台中显示解码的协议信息.参考Tracer javadocs 来了解详细信息.

 

3.2 Logging Subsystem AMQP Appenders

框架为多个流行的日志系统提供了日志appenders:

  • log4j (从Spring AMQP1.1版本开始)
  • logback (从Spring AMQP1.4版本开始)
  • log4j2 (从Spring AMQP1.6版本开始)

appenders使用正常机制为为子系统配置,可用属性参照下面的规定。

3.2.1 共同属性

下面的属性对于所有appenders都可用:

Table 3.4. 共同Appender属性

 

Property Default Description
exchangeName
logs

用于发布日志事件的交换器名称.

exchangeType
topic

发布日志事件的交换器类型- 只在appender声明了交换器的情况下才需要. 参考declareExchange.

routingKeyPattern
%c.%p

日志子系统生成路由键的模式格式.

applicationId

 

Application ID - 如果模式包含 %X{applicationId},则将其添加到路由键.

senderPoolSize
2

用于发布日志事件的线程数目.

maxSenderRetries
30

当broker不可用时或有某些错误时,重试的次数. 延时重试像: N ^ log(N)N 表示重试次数.

addresses

 

一个逗号分隔的broker地址列表: host:port[,host:port]* -覆盖host 和 port.

host
localhost

要连接RabbitMQ的主机.

port
5672
 
virtualHost
/

要连接的RabbitMQ虚拟主机.

username
guest

要连接RabbitMQ的用户.

password
guest

要连接RabbitMQ的用户密码.

contentType
text/plain

日志消息的content-type属性

contentEncoding

 

 日志消息的content-encoding属性.

declareExchange
false

当appender启动时,是否需要声明配置的交换器.也可参考 durable 和autoDelete.

durable
true

declareExchange 为 true ,durable 标志才会设置此值.

autoDelete
false

当 declareExchange 为true , auto delete 标志才会设置此值.

charset
null

当将字符串转成byte[]时要使用的编码,默认为null (使用系统默认字符集).如果当前平台上不支持此字符集,将回退到使用系统字符集.

deliveryMode
PERSISTENT

PERSISTENT 或 NON_PERSISTENT 用于决定RabbitMQ是否应该持久化消息.

generateId
false

用于确定messageId 属性是否需要设置成唯一值.

clientConnectionProperties
null

一个逗号分隔的key:value 对,它是连接RabbitMQ时设置的自定义客户端属性

3.2.2 Log4j Appender

样例log4j.properties片断. 

log4j.appender.amqp.addresses=foo:5672,bar:5672
log4j.appender.amqp=org.springframework.amqp.rabbit.log4j.AmqpAppender
log4j.appender.amqp.applicationId=myApplication
log4j.appender.amqp.routingKeyPattern=%X{applicationId}.%c.%p
log4j.appender.amqp.layout=org.apache.log4j.PatternLayout
log4j.appender.amqp.layout.ConversionPattern=%d %p %t [%c] - <%m>%n
log4j.appender.amqp.generateId=true
log4j.appender.amqp.charset=UTF-8
log4j.appender.amqp.durable=false
log4j.appender.amqp.deliveryMode=NON_PERSISTENT
log4j.appender.amqp.declareExchange=true

3.2.3 Log4j2 Appender

样例 log4j2.xml 片断


    ...
    
    

3.2.4 Logback Appender

样例 logback.xml 片断


    
        %n ]]>
    
    foo:5672,bar:5672
    36
    myApplication
    %property{applicationId}.%c.%p
    true
    UTF-8
    false
    NON_PERSISTENT
    true

3.2.5 定制Messages

每个appenders都可以子类化,以允许你在发布前修改消息.

Customizing the Log Messages. 

public class MyEnhancedAppender extends AmqpAppender {

    @Override
 public Message postProcessMessageBeforeSend(Message message, Event event) {
        message.getMessageProperties().setHeader("foo", "bar");
        return message;
    }

}

3.2.6 定制客户端属性

简化 String 属性

每个appender都支持在RabbitMQ连接中添加客户端属性.

log4j. 

log4j.appender.amqp.clientConnectionProperties=foo:bar,baz:qux

logback. 

...
foo:bar,baz:qux
...

log4j2. 


    ...
    

这些属性是逗号分隔的key:value 队列表; 键和值不能包含逗号或 冒号.

当RabbitMQ Admin UI中查看连接上,你会看到这些属性.

 

Log4j和logback先进技术

使用 log4j 和 logback appenders, appenders 可以是子类化的, 允许你在连接建立前,修改客户连接属性:

定制客户端连接属性. 

public class MyEnhancedAppender extends AmqpAppender {

    private String foo;

    @Override
protected void updateConnectionClientProperties(Map clientProperties) {
        clientProperties.put("foo", this.foo);
    }

    public void setFoo(String foo) {
        this.foo = foo;
    }

}

对于 log4j2, 添加 log4j.appender.amqp.foo=bar 到log4j.properties 来设置发展.

对于logback, 在logback.xml中添加 bar .

当然,对于像这个例子中简单的String 属性,可以使用先前的技术;

子类允许更丰富的属性(如添加 Map 的numeric 属性).

使用log4j2, 子类是不被支持的,因为 log4j2 使用静态工厂方法.

3.3 样例应用程序

3.3.1 介绍

 Spring AMQP Samples 项目包含了两个样例应用程序. 第一个简单的"Hello World" 示例演示了同步和异步消息的处理. 它为理解基础部分提供了一个很好的开端.
第二个基于股票交易的例子演示了真实应用程序中的交互场景.在本章中,我们会每个示例进行快速浏览,使您可以专注于最重要的组成部分.
这两个例子都是基于Maven的,因此你可以直接将它们导入任何支持Maven的IDE中(如. SpringSource Tool Suite).

3.3.2 Hello World

介绍

Hello World示例演示了同步和异步消息处理.你可以导入spring-rabbit-helloworld 示例到IDE中并跟随下面的讨论.

同步例子

src/main/java 目录中,导航到org.springframework.amqp.helloworld 包中.

打开HelloWorldConfiguration 类,你可以注意到它包含了@Configuration 类级注解和一些@Bean 方法级注解. 

这是Spring 的基于Java的配置.你可进一步的了解here.

@Bean
public ConnectionFactory connectionFactory() {
    CachingConnectionFactory connectionFactory =null;
    connectionFactory =new CachingConnectionFactory("localhost");
    connectionFactory.setUsername("guest");
    connectionFactory.setPassword("guest");
    return connectionFactory;
}

配置中同样也包含了RabbitAdmin的实例, 它会默认查找类型为Exchange, Queue, 或 Binding的bean并在broker中进行声明.
事实上,"helloWorldQueue" bean是在HelloWorldConfiguration 中生成的,因为它是 Queue的实例.

@Bean
public Queue helloWorldQueue() {
    returnnew Queue(this.helloWorldQueueName);
}

重看"rabbitTemplate"bean配置,你会看到它将helloWorldQueue的名称设成了"queue"属性(用于接收消息) 以及"routingKey" 属性(用于发送消息).

现在,我们已经探索了配置,让我们看看实际上使用这些组件的代码。

首先,从同一个包内打开Producer类。它包含一个用于创建Spring ApplicationContext的main()方法.

 

publicstaticvoid main(String[] args) {

ApplicationContext context =
        new AnnotationConfigApplicationContext(HelloWorldConfiguration.class);
    AmqpTemplate amqpTemplate = context.getBean(AmqpTemplate.class);
    amqpTemplate.convertAndSend("Hello World");
    System.out.println("Sent: Hello World");
}

在上面的例子中你可以看到, 取回的AmqpTemplate用来发送消息.因为客户端代码应该尽可能地依赖于接口,因此类型是AmqpTemplate而不是RabbitTemplate. 
即使在HelloWorldConfiguration中创建的bean是RabbitTemplate的实例,依赖于接口则意味着这端代码更具有便携性(portable) (配置可以独立于代码进行修改).
因为convertAndSend() 方法是通过模板来调用的,因此模板会将调用委派给它的MessageConverter实例.在这种情况下,它默认使用的是SimpleMessageConverter,但也可以在HelloWorldConfiguration中为"rabbitTemplate"指定其它的实现.

现在打开Consumer类. 它实际上共享了同一个配置基类,这意味着它将共享"rabbitTemplate" bean. 这就是为什么我们要使用"routingKey" (发送) 和"queue" (接收)来配置模板的原因. 
正如你在Section 3.1.4, “AmqpTemplate”中看到的,你可以代替在发送方法中传递routingKey参数,代替在接收方法中传递queue 参数.  Consumer 代码基本上是Producer的镜子,只不过调用的是receiveAndConvert()而非convertAndSend()方法.

publicstaticvoid main(String[] args) {
    ApplicationContext context =
        new AnnotationConfigApplicationContext(RabbitConfiguration.class);
    AmqpTemplate amqpTemplate = context.getBean(AmqpTemplate.class);
    System.out.println("Received: " + amqpTemplate.receiveAndConvert());
}

你如果运行Producer,然后再运行Consumer, 在控制台输出中,你应该能看到消息"Received: Hello World" 

异步示例

我们已经讲解了同步Hello World样例, 是时候移动到一个稍微先进,但更强大的选择上了.稍微修改一下代码,Hello World 样例就可以可以提供异步接收的示例了,又名 Message-driven POJOs. 事实上,有一个子包明确地提供了这种功能: org.springframework.amqp.samples.helloworld.async.

再一次地我们将从发送端开始. 打开ProducerConfiguration类可注意到它创建了一个"connectionFactory"和"rabbitTemplate" bean. 
这次,由于配置是专用于消息发送端,因此我们不需要任何队列定义,RabbitTemplate只须设置routingKey属性. 
回想一下,消息是发送到交换器上的而不是直接发到队列上的. AMQP默认交换器是无名称的direct类型交换器.
所有队列都是通过使用它们的名称作为路由键绑定到默认交换器上的.这就是为什么在这里我们只提供路由键的原因.

public RabbitTemplate rabbitTemplate() {
    RabbitTemplate template = new RabbitTemplate(connectionFactory());
    template.setRoutingKey(this.helloWorldQueueName);
    return template;
}

由于这个示例展示的是异步消息处理,生产方设计为连续发送消息(尽管类似于同步版本中的 message-per-execution模型,但不太明显,实际上它是消息驱动消费者)负责连续发送消息的组件是作为ProducerConfiguration类中的内部类来定义的,每3秒执行一次.

static class ScheduledProducer {

   @Autowired
 private volatile RabbitTemplate rabbitTemplate;

    private final AtomicInteger counter = new AtomicInteger();

    @Scheduled(fixedRate = 3000)
 public void sendMessage() {
        rabbitTemplate.convertAndSend("Hello World " + counter.incrementAndGet());
    }
}

你不必要完全了解这些细节,因为真正的关注点是接收方(我们马上就会讲解).然而,如果你还熟悉Spring 3.0 任务调度支持,你可从here这里来了解. 
简短故事是:在 ProducerConfiguration 中的"postProcessor" bean使用调度器来注册了任务.

现在,让我们转向接收方. 为强调 Message-driven POJO 行为,将从对消息起反应的组件开始. 

此类被称为HelloWorldHandler.

publicclass HelloWorldHandler {

    publicvoid handleMessage(String text) {
        System.out.println("Received: " + text);
    }

}

相当明显的, 这是一个POJO. 它没有继承任何基类,它没有实现任何接口,它甚至不包含任何导入. 它将通过Spring AMQP MessageListenerAdapter来适配MessageListener接口.然后适配器可配置在SimpleMessageListenerContainer上.
在这个例子中,容器是在ConsumerConfiguration类中创建的.你可以看到POJO是包装在适配器中的.

@Bean
public SimpleMessageListenerContainer listenerContainer() {
    SimpleMessageListenerContainer container = new SimpleMessageListenerContainer();
    container.setConnectionFactory(connectionFactory());
    container.setQueueName(this.helloWorldQueueName);
    container.setMessageListener(new MessageListenerAdapter(new HelloWorldHandler()));
    return container;
}

SimpleMessageListenerContainer是一个Spring生命周期组件,默认会自动启动.如果你看了Consumer类的内部,你会看到main()方法中除了一行启动创建ApplicationContext的代码外,其它什么都没有. 
Producer的main()方法也只有一行启动,因为以 @Scheduled注解的组件会自动开始执行.你可以任何顺序来启动Producer 和Consumer,你会看每秒就会发送消息和接收到消息.

3.3.3 股票交易(Stock Trading)

Stock Trading 示例演示了比Hello World示例更高级的消息场景.然而,配置却是很相似的 - 只是有一点复杂.
由于我们已经详细讲解了Hello World配置,因此在这里我们将重点关注不一样的东西. 有一个服务器发送市场数据(股票报价)到Topic交换器中. 
然后,客户端可订阅市场数据,即通过使用路由模式(如. "app.stock.quotes.nasdaq.*")来绑定队列(e.g. "app.stock.quotes.nasdaq.*"). 
这个例子的其它主要功能是 有一个请求回复“股票交易”的互动,它是由客户发起并由服务器来处理的. 这涉及到一个私有的“回复(replyTo)”队列,发送客户端的信息在请求消息中。

服务器的核心配置在RabbitServerConfiguration类中(位于 org.springframework.amqp.rabbit.stocks.config.server 包中).
它继承了 AbstractStockAppRabbitConfiguration. 这是服务器和客户端定义常用资源的地方,包括市场数据Topic交换器(其名称为app.stock.marketdata) 以及服务器公开股票交易的队列(其名称为app.stock.request). 
在那个公共配置文件中,你会看到在RabbitTemplate上配置了一个JsonMessageConverter.

服务器特有配置由2部分组成.首先,它在RabbitTemplate上配置了市场数据交换器,这样在发送消息时,就不必提供交换器名称.它是通过基础配置类中的抽象回调方法中定义做到这一点的.

public void configureRabbitTemplate(RabbitTemplate rabbitTemplate) {
    rabbitTemplate.setExchange(MARKET_DATA_EXCHANGE_NAME);
}

其次, 声明了股票请求队列.在这里,它不需要任何明确的绑定,因为它将以它自己的名称作为路由键来绑定到无名称的默认交换器上.正如先前提到的,AMQP规范定义了此种行为.

@Beanpublic Queue stockRequestQueue() {
    returnnew Queue(STOCK_REQUEST_QUEUE_NAME);
}

现在你已经看过了服务器的AMQP资源配置,导航到src/test/java目录下的org.springframework.amqp.rabbit.stocks包.在那里你会实际的 提供了main()方法的Server类.
它基于server-bootstrap.xml 创建了一个ApplicationContext.在那里,你会看到发布虚假市场数据的调度任务.

那个配置依赖于Spring 3.0的"task"命名空间支持.bootstrap配置文件也导入了其它一些文件.最令人关注的是位于src/main/resources目录下的server-messaging.xml.在那里,你会看到"messageListenerContainer" bean,它负责处理股票交易请求. 
最后在看一下定义在src/main/resources目录下的server-handlers.xml,其中定义了一个 "serverHandler" bean.这个bean是ServerHandler类的实例,它是Message-driven POJO 的好例子,它也有发送回复消息的能力.
注意,它自身并没有与框架或任何AMQP概念耦合.它只是简单地接受TradeRequest并返回一个TradeResponse.

public TradeResponse handleMessage(TradeRequest tradeRequest) { ...
}

现在我们已经看了服务端的重要配置和代码,让我们转向客户端.最佳起点是从 org.springframework.amqp.rabbit.stocks.config.client 包下的RabbitClientConfiguration开始. 

注意,它声明了两个不带明确参数的队列.

@Bean
public Queue marketDataQueue() {
    return amqpAdmin().declareQueue();
}

@Bean
public Queue traderJoeQueue() {
    return amqpAdmin().declareQueue();
}

那些是私有队列, 唯一名称会自动自成.客户端会用第一个生成的队列来绑定由服务端公开的市场交换器. 
记住在AMQP中,消费者与队列交互,而生产者与交换器交互. 队列和交换器之间的绑定指示broker从给定的交换器中投递或路由什么消息给队列.
由于市场交换器是一个Topic交换器,绑定可通过路由正则表达式来表达. 

RabbitClientConfiguration声明了一个Binding对象,其对象是通过BindingBuilder的便利API来生成的.

@Value("${stocks.quote.pattern}")
private String marketDataRoutingKey;

@Bean
public Binding marketDataBinding() {
    return BindingBuilder.bind(
        marketDataQueue()).to(marketDataExchange()).with(marketDataRoutingKey);
}

注意,实际值已经在属性文件(src/main/resources目录下的"client.properties")中外部化了,因此我们使用Spring的@Value 注解来注入值.这通常是一个好主意,否则值就会硬编码在类中,没有修改就没有重新编译.

在这种情况下,通过修改绑定中的路由正则表达式,可很容易地运行多个版本的Client.让我们立即尝试.

启动运行org.springframework.amqp.rabbit.stocks.Server然后再运行 org.springframework.amqp.rabbit.stocks.Client.你将会看到NASDAQ股票的交易报价,因为关联stocks.quote.pattern 键的值在client.properties中是app.stock.quotes.nasdaq.
现在,保持现有Server 和Client 运行,将其属性值修改为app.stock.quotes.nyse.再启动第二个
Client实例.你会看到第一个client仍然接收NASDAQ 报价,而第二个client接收的NYSE报价. 你可以改变模式,获取所有的股票报价或个别股票的报价。

最后一个我们将暴露的特性是从客户端的角度来看待请求-回复交互.记住我们已经看了ServerHandler,它会接受TradeRequest对象并返回TradeResponse对象. 客户端相应的代码是 RabbitStockServiceGateway(位于org.springframework.amqp.rabbit.stocks.gateway 包).为发送消息,它会委派给RabbitTemplate.

public void send(TradeRequest tradeRequest) {
    getRabbitTemplate().convertAndSend(tradeRequest, new MessagePostProcessor() {
        public Message postProcessMessage(Message message) throws AmqpException {
            message.getMessageProperties().setReplyTo(new Address(defaultReplyToQueue));
            try {
                message.getMessageProperties().setCorrelationId(
                    UUID.randomUUID().toString().getBytes("UTF-8"));
            }
            catch (UnsupportedEncodingException e) {
                thrownew AmqpException(e);
            }
            return message;
        }
    });
}

注意,在发送消息前,它设置了"replyTo"地址. 这提供了队列,此队列是由上面的"traderJoeQueue" bean 定义生成的. 以下是StockServiceGateway类的@Bean定义.

@Bean
public StockServiceGateway stockServiceGateway() {
    RabbitStockServiceGateway gateway = new RabbitStockServiceGateway();
    gateway.setRabbitTemplate(rabbitTemplate());
    gateway.setDefaultReplyToQueue(traderJoeQueue());
    return gateway;
}

如果你没有运行服务器和客户端,现在就启动它们. 尝试使用100 TCKR的格式来发送请求.经过一个简短的人工延迟来模拟“处理”请求,你应该看到一个确认消息出现在客户端上。

3.4 测试支持

3.4.1 介绍

为异步程序写集成测试比测试简单程序更复杂. 当引入了@RabbitListener这样的注解时,这尤其更加复杂.

现在的问题是发送消息后,如何来验证, 监听器按预期收到了消息.

框架自身带有许多单元测试和集成测试;有些使用mocks, 另外一些使用真实的RabbitMQ broker来集成测试. 您可以参照测试场景的一些想法进行测试。

Spring AMQP 1.6版本引入了sring-rabbit-test jar ,它提供一些测试复杂场景的测试. 预计这一项目将随着时间的推移进行扩展,但我们需要社会反馈以帮助测试。请使用JIRA问题或GitHub提供这样的反馈。

3.4.2 Mockito Answer 实现

当前有两个Answer 实现可帮助测试:

第一个, LatchCountDownAndCallRealMethodAnswer 提供了返回null和计数下一个锁存器的Answer.

LatchCountDownAndCallRealMethodAnswer answer = new LatchCountDownAndCallRealMethodAnswer(2);
doAnswer(answer)
    .when(listener).foo(anyString(), anyString());

...

assertTrue(answer.getLatch().await(10, TimeUnit.SECONDS));

第二个, LambdaAnswer 提供了一种调用真正方法的机制,并提供机会来返回定制结果(基于InvocationOnMock和结果).

public class Foo {

    public String foo(String foo) {
        return foo.toUpperCase();
    }

}
Foo foo = spy(new Foo());

doAnswer(new LambdaAnswer(true, (i, r) -> r + r))
    .when(foo).foo(anyString());
assertEquals("FOOFOO", foo.foo("foo"));

doAnswer(new LambdaAnswer(true, (i, r) -> r + i.getArguments()[0]))
    .when(foo).foo(anyString());
assertEquals("FOOfoo", foo.foo("foo"));

doAnswer(new LambdaAnswer(false, (i, r) ->
    "" + i.getArguments()[0] + i.getArguments()[0])).when(foo).foo(anyString());
assertEquals("foofoo", foo.foo("foo"));

When using Java 7 or earlier:

doAnswer(new LambdaAnswer(true, new ValueToReturn() {
    @Overridepublic String apply(InvocationOnMock i, String r) {
        return r + r;
    }
})).when(foo).foo(anyString());

3.4.3 @RabbitListenerTest and RabbitListenerTestHarness

在你的@Configuration 类中使用 @RabbitListenerTest (它也会通过@EnableRabbit来启用@RabbitListener 探测).注解会导致框架使用子类RabbitListenerTestHarness来代替标准RabbitListenerAnnotationBeanPostProcessor.

RabbitListenerTestHarness 通过两种方式来增强监听器 - 将其包装进Mockito Spy, 启用了Mockito 存根和验证操作.也可在监听器添加Advice 来启用对参数,结果或异常的访问.
你可以控制哪一个(或两个)来在@RabbitListenerTest上启用属性. 后者用于访问调用中更为低级数据- 它也支持测试线程阻塞,直到异步监听器被调用.

重要

final @RabbitListener 不能被发现或通知 ,同时,只有带id属性的监听器才能发现或通知.

让我们看一些例子.

使用spy:

@Configuration
@RabbitListenerTest
public class Config {

    @Bean
 public Listener listener() {
        returnnew Listener();
    }

    ...

}

public class Listener {

    @RabbitListener(id="foo", queues="#{queue1.name}")
 public String foo(String foo) {
        return foo.toUpperCase();
    }

    @RabbitListener(id="bar", queues="#{queue2.name}")
 public void foo(@Payload String foo, @Header("amqp_receivedRoutingKey") String rk) {
        ...
    }

}

public class MyTests {

    @Autowired
private RabbitListenerTestHarness harness;
@Test
 public void testTwoWay() throws Exception {
        assertEquals("FOO", this.rabbitTemplate.convertSendAndReceive(this.queue1.getName(), "foo"    ));

        Listener listener = this.harness.getSpy("foo"); 
        assertNotNull(listener);
        verify(listener).foo("foo");
    }

    @Test
public void testOneWay() throws Exception {
        Listener listener = this.harness.getSpy("bar");
        assertNotNull(listener);

        LatchCountDownAndCallRealMethodAnswer answer = new LatchCountDownAndCallRealMethodAnswer(2); 
        doAnswer(answer).when(listener).foo(anyString(), anyString());
  this.rabbitTemplate.convertAndSend(this.queue2.getName(), "bar");
  this.rabbitTemplate.convertAndSend(this.queue2.getName(), "baz");

        assertTrue(answer.getLatch().await(10, TimeUnit.SECONDS));
        verify(listener).foo("bar", this.queue2.getName());
        verify(listener).foo("baz", this.queue2.getName());
    }

}

将harness 注入进测试用于,这样我们可访问spy.

获取spy引用,这样我们可以验证是否按预期在调用. 由于这是一个发送和接收操作,因此不必暂停测试线程,因为RabbitTemplate 在等待回复时已经暂停过了.

在这种情况下,我们只使用了发送操作,因为我们需要一个门闩来等待对容器线程中监听器的异步调用. 我们使用了Answer 一个实现来帮助完成.

配置spy来调用Answer.

使用捕获建议:

@Configuration
@ComponentScan
@RabbitListenerTest(spy = false, capture = true)
public class Config {

}

@Service
public class Listener {

    private boolean failed;

    @RabbitListener(id="foo", queues="#{queue1.name}")
 public String foo(String foo) {
        return foo.toUpperCase();
    }

    @RabbitListener(id="bar", queues="#{queue2.name}")
public void foo(@Payload String foo, @Header("amqp_receivedRoutingKey") String rk) {
        if (!failed && foo.equals("ex")) {
            failed = true;
            thrownew RuntimeException(foo);
        }
        failed = false;
    }

}

public class MyTests {

     @Autowired
  private RabbitListenerTestHarness harness;
  @Test
  public void testTwoWay() throws Exception {
        assertEquals("FOO", this.rabbitTemplate.convertSendAndReceive(this.queue1.getName(), "foo"  ));

        InvocationData invocationData =
            this.harness.getNextInvocationDataFor("foo", 0, TimeUnit.SECONDS); 
        assertThat(invocationData.getArguments()[0], equalTo("foo"));     
        assertThat((String) invocationData.getResult(), equalTo("FOO"));
    }

    @Test
  public void testOneWay() throws Exception {
        this.rabbitTemplate.convertAndSend(this.queue2.getName(), "bar");
        this.rabbitTemplate.convertAndSend(this.queue2.getName(), "baz");
        this.rabbitTemplate.convertAndSend(this.queue2.getName(), "ex");

        InvocationData invocationData =
            this.harness.getNextInvocationDataFor("bar", 10, TimeUnit.SECONDS); 
        Object[] args = invocationData.getArguments();
        assertThat((String) args[0], equalTo("bar"));
        assertThat((String) args[1], equalTo(queue2.getName()));

        invocationData = this.harness.getNextInvocationDataFor("bar", 10, TimeUnit.SECONDS);
        args = invocationData.getArguments();
        assertThat((String) args[0], equalTo("baz"));

        invocationData = this.harness.getNextInvocationDataFor("bar", 10, TimeUnit.SECONDS);
        args = invocationData.getArguments();
        assertThat((String) args[0], equalTo("ex"));
        assertEquals("ex", invocationData.getThrowable().getMessage()); 
    }

}

将harness注入进测试用例,以便我们能获取spy的访问.

使用 harness.getNextInvocationDataFor() 来获取调用数据 - 在这种情况下,由于它处于request/reply 场景,因为没有必要等待,因为测试线程在RabbitTemplate 中等待结果的时候,已经暂停过了.

我们可以验证参数和结果是否与预期一致

这次,我们需要时间来等待数据,因为它在容器线程上是异步操作,我们需要暂停测试线程.

当监听器抛出异常时,可用调用数据中的throwable 属性

 

4. Spring 整合- 参考

这部分参考文档提供了在Spring集成项目中提供AMQP支持的快速介绍.

4.1 Spring 整合AMQP支持4.1.1 介绍

Spring Integration 项目包含了构建于Spring AMQP项目之上的AMQP 通道适配器(Channel Adapters)和网关(Gateways). 那些适配器是在Spring集成项目中开发和发布的.在Spring 集成中, "通道适配器" 是单向的,而网关是双向的(请求-响应).
我们提供了入站通道适配器(inbound-channel-adapter),出站通道适配器( outbound-channel-adapter), 入站网关(inbound-gateway),以及出站网关(outbound-gateway).

由于AMQP 适配器只是Spring集成版本的一部分,因为文档也只针对Spring集成发行版本部分可用. 

作为一个品酒师,我们只快速了解这里的主要特征。

4.1.2 入站通道适配器

为了从队列中接收AMQP消息,需要配置一个个

4.1.3 出站通道适配器
为了向交换器发送AMQP消息,需要配置一个. 除了交换名称外,还可选择提供路由键。

4.1.4 入站网关

为了从队列中接收AMQP消息,并回复到它的reply-to地址,需要配置一个.

4.1.5 出站网关

为了向交换器发送AMQP消息并接收来自远程客户端的响应,需要配置一个.

除了交换名称外,还可选择提供路由键。

 

5. 其它资源

除了这份参考文档,还有其它资源可帮助你了解AMQP.

5.1 进阶阅读

对于那些不熟悉AMQP的人来说,  规范 实际上具有相当的可读性. 

转载于:https://my.oschina.net/jwl210/blog/810646

你可能感兴趣的:(Spring AMQP 1.6完整参考指南)