RabbitMQ实战:消息通信模式和最佳实践

本系列是「RabbitMQ实战:高效部署分布式消息队列」书籍的总结笔记。

关注我的微信公众号:情情说,分享工作、学习和生活 ~

通过前2篇的介绍,了解了消息通信的主要元素和交互过程,以及如何运行和管理RabbitMQ,这篇将站在开发模式的角度理解「面向消息通信」带来的好处,以及在各种场景下的最佳实践。

通过介绍,你会了解到:

  • 面向消息通信的好处
  • 发后即忘模型
  • 用RabbitMQ实现RPC

面向消息通信的好处

主要从异步状态思维、处理能力扩展性、集成复杂度方面,说明面向消息通信的好处。

异步状态思维

当将消息通信集成到应用程序时,开发模式将从同步模型变为异步模型,RabbitMQ提供了不同的方法,允许我们在一处发送请求,在另一处进行处理,这样同步程序可以继续执行其他逻辑。

举个简单的例子来说明,通过支付宝还信用卡:

  • 用户填写信用卡号、发卡银行、持卡人姓名、还款金额,提交还款申请;
  • 支付宝会立即提示用户,申请已提交,多少小时内完成还款;
  • 还款完成后,会推送给用户一条消息,提醒还款是否成功;

如果是同步请求,用户需要等待几个小时查看结果,等待过程中不能进行其他操作,这是很不合理的。

异步的思维是将请求和处理分离,在应用中紧密耦合的两部分中间使用RabbitMQ,请求解析后,发送一条业务能够理解的消息到RabbitMQ,就返回给用户,真正的处理由另外的服务异步处理。

扩展性

随着业务的扩展,对服务处理能力的要求越来越高,RabbitMQ可以很简单的增加处理能力。

因为RabbitMQ可以将请求在处理服务器间平均地分发,不需要负载均衡器了。

零成本API

系统间相互调用,需要约定一套API,通常来讲,需要花费一点点时间,编写一大段代码将传入的HTTP请求转化为应用程序中的函数调用。

如果使用AMQP来连接应用程序的各个部分,无需额外定义API,使用消息通信即可。另外, AMQP是语言无关的,拥有数十种语言的本地语言绑定。

发后即忘模型

当考虑消息通信能够解决的问题类型时,消息通信适用的主要领域是的「发后即忘」处理模式。关心的是任务将会完成,但无须实时完成,创建一个任务,发送到交换器上后,就可以返回继续工作,甚至都不需要通知用户任务已经完成。

匹配该模式的两种类型任务:

  • 批处理:针对大型数据集合的工作或者转换,多个任务对数据集合的独立部分进行操作;
  • 通知:对发送事件的描述,可以是消息的日志,或者通知另一个程序或者管理员;

书上介绍的实例比较简单,就不在此列出了,主要是根据不同的场景,确定交换器的类型和routingkey,可以参考上一篇介绍的「收集日志」的例子进行理解。

用RabbitMQ实现RPC

有多种方式来实现远程过程调用RPC,比如REST API、SOAP、Thrift等,这些传统的RPC实现方法有共同之处:客户端和服务器紧密相连、而且要等待返回结果。另外考虑这些问题:

  • 当有多个服务节点时,客户端如何发现对应服务器;
  • 如果客户端连接的RPC服务器崩溃了,客户端需要额外逻辑进行重连;

通过MQ服务器来实现时,只是简单地发布消息而已,将消息路由到合适的地方放,通过多台RPC服务器对消息进行负载均衡,当处理消息的服务器崩溃时,将RPC消息重发到另一台。

现在的问题在于,如果将应答返回给客户端?

RabbitMQ使用消息来发回应答,在AMQP消息头里有一个字段叫做reply_to,消息的生成者可以通过该字段来确定队列名称,并监听队列等待应答,消息接收者能够检查reply_to字段,并创建包含应答内容的新的消息,并以队列名称作为路由键。

关于reply_to的队列名称,如果生成者声明了没有名字的队列,RabbitMQ为自动生成一个唯一的队列名,同时在声明的时候指定exclusive参数,确保只有创建队列的生产者可以读取队列上的消息。

这样,所有RPC客户端要做的,就是声明临时的、排他的、匿名队列,并将该队列名称包含到RPC消息的reply_to头中,这样服务器端就知道应答消息该发往哪儿了。

很多场景使用「发后即忘」模型,不需要处理者响应,如果需要响应,可以使用RabbitMQ的RPC模型。

下一篇将介绍RabbitMQ集群和高可用性以及它们的设置。

欢迎扫描下方二维码,关注我的个人微信公众号 ~


RabbitMQ实战:消息通信模式和最佳实践

相关推荐