【黑马SpringCloud(4)】RabbitMQ异步通信实现

本文介绍了异步通信在微服务中的重要性,对比了同步和异步通信的优缺点。接着详细讲解了消息队列(MQ)的概念,包括RabbitMQ中的不同消息模型:无交换机的基本消息队列、工作消息队列,以及有交换机的发布订阅模型(FanoutExchange、DirectExchange、TopicExchange)。此外,还展示了如何使用SpringAMQP实现消息发送和接收,以及工作消息队列的处理。最后提到了消息转换器,讨论了JDK序列化的问题和JSON转换器的使用。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

代码地址:https://gitee.com/suisui9857/mq-demo

初识MQ

微服务间通讯有同步和异步两种方式:

同步通讯: 时效性较强,需要实时响应,可以立即得到结果。(打电话)

缺点:

  • 耦合度高:每次加入新的需求,都要修改原来的代码。
  • 性能和吞吐能力下降:调用者需要等待服务提供者响应,如果调用链过长则响应时间等于每次调用的时间之和。
  • 有额外的资源消耗:调用链中的每个服务在等待响应过程中,不能释放请求占用的资源,高并发场景下会极度浪费系统资源。
  • 有级联失败问题:如果服务提供者出现问题,所有调用方都会跟着出现问题,迅速导致整个微服务群故障。

异步通讯: 就像发邮件,不需要马上回复。
用户支付后需要调用订单服务完成订单状态修改,调用物流服务,从仓库分配响应的库存并准备发货。

在事件模式中,支付服务是事件发布者(publisher),在支付完成后只需要发布一个支付成功的事件(event),事件中带上订单id。订单服务和物流服务是事件订阅者(Consumer),订阅支付成功的事件,监听到事件后完成自己业务即可。

为了解除事件发布者与订阅者之间的耦合,两者并不是直接通信,而是有一个中间人(Broker)。发布者发布事件到Broker,不关心谁来订阅事件。订阅者从Broker订阅事件,不关心谁发来的消息。Broker 是一个像数据总线一样的东西,所有的服务要接收数据和发送数据都发到这个总线上,这个总线就像协议一样,让服务间的通讯变得标准和可控。
在这里插入图片描述
优点:

  • 吞吐量提升:无需等待订阅者处理完成,响应更快速
  • 故障隔离:服务没有直接调用,不存在级联失败问题
  • 调用间没有阻塞,不会造成无效的资源占用
  • 耦合度极低,每个服务都可以灵活插拔,可替换,不需要调用
  • 流量削峰:不管发布事件的流量波动多大,都由Broker接收,订阅者可以按照自己的速度去处理事件

缺点:

  • 架构复杂了,业务没有明显的流程线,不好管理
  • 需要依赖于Broker的可靠、安全、性能

MQ,中文是消息队列(MessageQueue),字面来看就是存放消息的队列。也就是事件驱动架构中的Broker。
几种常见MQ的对比:

RabbitMQ ActiveMQ RocketMQ Kafka
公司/社区 Rabbit Apache 阿里 Apache
开发语言 Erlang Java Java Scala&Java
协议支持 AMQP,XMPP,SMTP,STOMP OpenWire,STOMP,REST,XMPP,AMQP 自定义协议 自定义协议
可用性 一般
单机吞吐量 一般 非常高
消息延迟 微秒级 毫秒级 毫秒级 毫秒以内
消息可靠性 一般 一般

追求可用性:Kafka、 RocketMQ 、RabbitMQ
追求可靠性:RabbitMQ、RocketMQ
追求吞吐能力:RocketMQ、Kafka
追求消息低延迟:RabbitMQ、Kafka

MQ的基本结构:
在这里插入图片描述

  • publisher:生产者
  • consumer:消费者
  • channel:通道,操作MQ的工具
  • exchange:交换机,路由消息到队列中
  • queue:队列,存储消息消息
  • virtual host:虚拟主机,对queue,exchange等资源的逻辑分组

常见消息模型

在线手册:https://www.rabbitmq.com/getstarted.html

无交换机:
基本消息队列(BasicQueue):
在这里插入图片描述
工作消息队列(WorkQueue):
在这里插入图片描述

有交换机:
发布订阅(Publish,Subscribe)根据交换机类型分为三种:
广播(Fanout Exchange):
在这里插入图片描述
路由(Direct Exchange):
在这里插入图片描述
主题(Topic Exchange):
在这里插入图片描述

基本消息队列演示:

publisher:消息发布者,将消息发送到队列queue
queue:消息队列,负责接收并缓存消息
consumer:订阅队列,处理队列中的消息
在这里插入图片描述
导入依赖:

<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-amqp</artifactId>
</dependency>

基本消息队列发送消息:
1.建立connection
2.创建channel
3.利用channel声明队列
4.利用channel向队列发送消息

public class PublisherTest {
   
    @Test
    public void testSendMessage() throws IOException, TimeoutException {
   
        // 1.建立连接
        ConnectionFactory factory = new ConnectionFactory();
        // 1.1.设置连接参数,分别是:主机名、端口号、vhost、用户名、密码
        factory.setHost("192.168.221.199");
        factory.setPort(5672);
        factory.setVirtualHost("/");
        factory.setUsername("admin");
        factory.setPassword("12345678");
        // 1.2.建立连接
        Connection connection = factory.newConnection();

        // 2.创建通道Channel
        Channel channel = connection.createChannel();

        // 3.创建队列
        String queueName = "simple.queue";
        channel.queueDeclare(queueName, false, false, false, null);

        // 4.发送消息
        String message = "hello, rabbitmq!";
        channel.basicPublish("", queueName, null, message.getBytes());
        System.out.println("发送消息成功:【" + message + "】");

        // 5.关闭通道和连接
        channel.close();
        connection.close();

    }
}

在这里插入图片描述

基本消息队列接收消息:
1.建立connection
2.创建channel
3.利用channel声明队列,都声明队列是避免队列不存在
4.定义consumer的消费行为handleDelivery()
5.利用channel将消费者与队列绑定

public class ConsumerTest {
   

    public static void main(String[] args) throws IOException, TimeoutException {
   
        // 1.建立连接
        ConnectionFactory factory = new ConnectionFactory();
        // 1.1.设置连接参数,分别是:主机名、端口号、vhost、用户名、密码
        factory.setHost("192.168.221.199");
        factory.setPort(5672);
        factory.setVirtualHost("/");
        factory.setUsername(
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值