【RabbitMQ】工作队列模式(使用案例)

文章目录


咱们在前面学习了简单模式的写法,接下来学习另外几种工作模式的写法。

快速入门程序就是简单模式。此处省略。

1. Work Queues(工作队列)

它其实是简单模式的增强版,和简单模式的区别就是:简单模式有一个消费者,工作队列模式支持多个消费者接收消息,消费者之间是竞争关系,每个消息只能被一个消费者接收。

![步骤:

2、引入依赖

先引入 rabbitmq 的依赖

xml 复制代码
<!-- Source: https://mvnrepository.com/artifact/com.rabbitmq/amqp-client -->
<dependency>
    <groupId>com.rabbitmq</groupId>
    <artifactId>amqp-client</artifactId>
    <version>5.20.0</version>
    <scope>compile</scope>
</dependency>

3、生产者代码编写

工作队列模式和简单模式区别是有多个消费者,所以生产者消费者代码差异不大。

首先,咱们把配置文件单独领出来,定义为常量。

java 复制代码
public class Constants {
    public static final String HOST = "8.156.77.78";
    public static final int PORT = 5672;
    public static final String USERNAME = "edison";
    public static final String PASSWORD = "edison";
    public static final String VIRTUAL_HOST = "my_app_vhost";

    // 工作队列模式
    public static final String WORK_QUEUE = "work.queue";
}

相比简单模式,生产者的代码基本一样,为了能看到多个消费者竞争的关系,我们一次发送10条消息。

我们把发送消息的地方,改为一次发送 10 条消息即可。

java 复制代码
package work;

import com.rabbitmq.client.Channel;
import com.rabbitmq.client.Connection;
import com.rabbitmq.client.ConnectionFactory;
import constant.Constants;

import java.io.IOException;
import java.util.concurrent.TimeoutException;

public class Producer {
    public static void main(String[] args) throws IOException, TimeoutException {
        // 1. 建立连接
        ConnectionFactory factory = new ConnectionFactory();
        factory.setHost(Constants.HOST);   // MQ所在的服务器地址
        factory.setPort(Constants.PORT);            // 端口号
        factory.setUsername(Constants.USERNAME);    // 账号
        factory.setPassword(Constants.PASSWORD);    // 密码
        factory.setVirtualHost(Constants.VIRTUAL_HOST);      // 虚拟主机
        Connection connection = factory.newConnection();

        // 2. 开启 channel 通道
        Channel channel = connection.createChannel();

        // 3. 声明交换机(使用内置的交换机即可)
        // ....

        // 4. 声明队列(如果队列不存在,则创建; 反之,存在就创建)
        channel.queueDeclare(Constants.WORK_QUEUE, true, false, false, null);

        // 5. 发送消息
        for (int i = 0; i < 10; i++) {
            String msg = "Hello work queue... " + i;
            channel.basicPublish("", Constants.WORK_QUEUE, null, msg.getBytes());
        }
        System.out.println("消息发生成功~~~");

        // 8. 资源释放
        channel.close();
        connection.close();
    }
}

4、消费者代码编写

消费者代码和简单模式一样,只是复制两份,两个消费者代码片段是一样的。

java 复制代码
package work;

import com.rabbitmq.client.*;
import constant.Constants;

import java.io.IOException;
import java.util.concurrent.TimeoutException;

public class Consumer1 {
    public static void main(String[] args) throws IOException, TimeoutException {
        // 1. 建立连接
        ConnectionFactory factory = new ConnectionFactory();
        factory.setHost(Constants.HOST);   // MQ所在的服务器地址
        factory.setPort(Constants.PORT);            // 端口号
        factory.setUsername(Constants.USERNAME);    // 账号
        factory.setPassword(Constants.PASSWORD);    // 密码
        factory.setVirtualHost(Constants.VIRTUAL_HOST);      // 虚拟主机
        Connection connection = factory.newConnection();

        // 2. 开启 channel 通道
        Channel channel = connection.createChannel();

        // 3. 声明交换机(使用内置的交换机即可)
        // ....

        // 4. 声明队列(如果队列不存在,则创建; 反之,存在就创建)
        channel.queueDeclare(Constants.WORK_QUEUE, true, false, false, null);

        // 5. 消费消息
        DefaultConsumer consumer = new DefaultConsumer(channel) {
            // 从队列中收到消息后, 就会执行的方法
            @Override
            public void handleDelivery(String consumerTag, Envelope envelope, AMQP.BasicProperties properties, byte[] body) throws IOException {
                // 收到消息以后就进行打印
                System.out.println("接收到消息: " + new String(body));
            }
        };
        channel.basicConsume(Constants.WORK_QUEUE, true, consumer);

        // 8. 资源释放
        //channel.close();
        //connection.close();
    }
}

5、观察结果

先启动两个消费者运行,再启动生产者。

为什么呢?如果先启动生产者,再启动消费者,由于消息较少,处理较快,那么第一个启动的消费者就会瞬间把 10 条消息消费掉,所以我们先启动两个消费者,再启动生产者。

先注释掉 Consumer 里面的【释放资源】的代码,然后分别启动 Consumer1 和 Consumer2,此时咱们的工作队列 work_queue 中就有两个消费者在等待

并且 Connections 的状态也是两个

然后启动 Producer 生产者的代码,让生产者发送 10 条消息到队列中去

此时 Consumer 1 收到的消息如下:

此时 Consumer 2 收到的消息如下:

并且在 RabbitMQ 的服务端上可以看到,此时队列里面是没有任何消息的,因为它被消费者瞬间就拿走了

相关推荐
MoFe12 小时前
【Rabbitmq】windows环境下搭建与使用
分布式·rabbitmq
qq_297574672 小时前
【Kafka系列·进阶第三篇】流处理与数据治理实战:Streams实时计算+Schema校验+多租户管控
分布式·kafka·linq
新缸中之脑3 小时前
Meta新模型Muse Spark上手体验
大数据·分布式·spark
Rick19933 小时前
Kafka 的 ISR 是什么
分布式·kafka
Thomas21433 小时前
pyspark 新接口 DataSource V2 写法 写入paimon为例
大数据·分布式·spark
REDcker4 小时前
RabbitMQ系列04 - 流控与信用机制
分布式·rabbitmq
謓泽4 小时前
2022年江西省大学生电子设计竞赛 H 题 — 分布式监控系统
分布式·电赛
rannn_1114 小时前
【Redis|高级篇2】多级缓存|JVM进程缓存、Lua语法、多级缓存实现(OpenResty)、缓存同步(Canal)
java·redis·分布式·后端·缓存·lua·openresty
Rick199317 小时前
Redis 分布式锁:核心使用场景
数据库·redis·分布式