【RabbitMQ】队列模型

news2025/4/18 2:20:56

1.概述

RabbitMQ作为消息队列,有6种队列模型,分别在不同的场景进行使用,分别是Hello World,Work queues,Publish/Subscribe,Routing,Topics,RPC。

下面就分别对几个模型进行讲述。

2.Hello World

这个模型也叫直连模型,从这个名字来看就知道它的原理很简单,是一个线性的且没有分支的模型。

生产者生产消息,把消息给队列,队列在把消息给消费者进行消费。

代码实现

工具类

public class RabbitMQUtils {
    public static Connection getConnection() throws IOException, TimeoutException {
        //创建一个连接工厂
        ConnectionFactory factory = new ConnectionFactory();
        factory.setHost("192.168.153.132");//设置rabbitmq的主机地址
        factory.setPort(5672);//设置rabbitmq的端口号
        factory.setUsername("admin");//设置用户名
        factory.setPassword("admin");//设置密码
        factory.setVirtualHost("/abc");//设置虚拟主机
        //根据连接工厂获取一个连接对象
        Connection connection = factory.newConnection();
        return connection;
    }
}

 生产者

/**
 * 基于HelloWorld消息模型的消息生产者
 */
public class MessageProvider1 {
    public static void main(String[] args) throws IOException, TimeoutException {
         Connection connection= RabbitMQUtils.getConnection();
        //根据连接对象创建一个Channel(信道)
        Channel channel = connection.createChannel();
        /**
         * 根据Channel声明一个队列
         * 参数1:队列的名称
         * 参数2:描述队列是否持久化(true 讲队列以文件的形式保存在硬盘上,下一次启动mq服务还可以看到该队列)
         * 参数3:是否独占队列(true 只有当前会话才能使用该队列)
         * 参数4:队列是否会自动删除
         * 参数5:对队列进行的额外设置
         */
        channel.queueDeclare("hello", true, false, false, null);
        /**
         * 发布消息
         * 参数1:交换机的名称
         * 参数2:队列的名称
         * 参数3:消息进行额外设置 MessageProperties.PERSISTENT_TEXT_PLAIN消息持久化
         * 参数4:消息主体,以字节数组的方式进行消息的发送
         */
        channel.basicPublish("","hello", MessageProperties.PERSISTENT_TEXT_PLAIN,"hello rabbitmq".getBytes());
        //关闭资源
        channel.close();
        connection.close();
    }
}

消费者部分 

/**
 * 基于HelloWorld消息模型的消息消费者
 */
public class MessageConsumer1 {
    public static void main(String[] args) throws IOException, TimeoutException {
            Connection connection= RabbitMQUtils.getConnection();
             Channel channel = connection.createChannel();
            //声明一个队列
            channel.queueDeclare("hello", true, false, false, null);
            /**
             * 消息的消费
             * 参数1:队列的名称
             * 参数2:是否开启自动确认机制 true 开启
             * 参数3:处理队列里面消息的回调函数
             */
             channel.basicConsume("hello",true,new DefaultConsumer(channel){
                     @Override
                     public void handleDelivery(String consumerTag, Envelope envelope, AMQP.BasicProperties properties, byte[] body) throws IOException {
                             String msg = new String(body);
                             System.out.println(msg);
                     }
             });

    }
}

 这时候先启动消费者等待生产者的消息,然后再开启生产者,那么消费者就会接收到消息,这里是”Hello rabbit“

这里的队列名称要自己再RabbitMQ的管理界面进行创建

在这个界面可以进行手动创建

3.Work queues

它与第一种模型不同的是,消息队列不再是单一的讲消息传送给一个消费者,而是可以传送给多个消费者。它就解决了生产者生产消息的速率大于消费者消费消息的速率,使消息不断地堆积在队列中。

同样的它的模型也很简单,也是生产消息给队列,队列再将消息分给队列。

代码实现

生产者

/**
 * 消息的生产者 -- 工作队列实现
 */
public class MessageProvider2 {
    public static void main(String[] args) throws Exception {
        Connection connection = RabbitMQUtils.getConnection();
        // 创建信道
        Channel channel = connection.createChannel();
        // 声明队列
        channel.queueDeclare("work", false, false, false, null);

        // 发送消息
        for (int i = 1; i <= 10; i++) {
            String message = "hello,SpringCloud" + i;
            channel.basicPublish("", "work", null, message.getBytes());
        }

        // 关闭资源
        channel.close();
        connection.close();
    }
}

消费者 

public class MessageConsumer_q2 {
    public static void main(String[] args) throws Exception {
        Connection connection = RabbitMQUtils.getConnection();
        final Channel channel = connection.createChannel();
        //声明队列
        channel.queueDeclare("work",false,false,false,null);
        //消费消息 参数2 false 关闭消息的自动确认机制
        channel.basicConsume("work",true,new DefaultConsumer(channel){
            @Override
            public void handleDelivery(String consumerTag, Envelope envelope,
                                       AMQP.BasicProperties properties, byte[] body) throws IOException {
                    System.out.println("消息是:" + new String(body));
            }
        });
    }
}

消费者的代码都一样的,就不重复写了。

但现在要思考一个问题,就是我这里的代码是开启了消息的自动确认机制,就是意思是队列会将消息按照顺序平均分配给消费者,也就是消费者会先再队列中确认消息(确认意味着消息从队列中删除),这样就会出现一个问题,有的消费者消费得快,有得消费者消费得慢,这样得结果显然不是我们想要的,我们肯定希望能者多劳,也就是按需分配,而不是按量平均分配 。还有一个问题,就是消费者一下子确认这么多消息然后慢慢消费,万一某个消费者在消费的时候挂掉了,这些消息都已经不在队列中了,那不就直接导致消息丢失了。

因此我们需要对消息进行手动确认而不是自动确认

public class MessageConsumer_q1 {
    public static void main(String[] args) throws Exception {
        Connection connection = RabbitMQUtils.getConnection();
        final Channel channel = connection.createChannel();
        //声明队列
        channel.queueDeclare("work", false, false, false, null);
        //消费消息  第二个参数改成false,即关闭自动确认
        channel.basicConsume("work", false, new DefaultConsumer(channel){
            @Override
            public void handleDelivery(String consumerTag, Envelope envelope, AMQP.BasicProperties properties, byte[] body) throws IOException {
                // 开启手动确认 不再同时确认多条消息
                channel.basicAck(envelope.getDeliveryTag(), false);
                System.out.println("消息是:" + new String(body));
            }
        });
    }
}

4.fanout

这个模型就相对复杂点,多了一个交换机。

广播类型的交换机可以和任意类型的队列进行配对。

每个消费者有自己的队列。

每个队列都要绑定到交换机.

生产者发送的消息,只能发送到交换机,交换机来决定要发给哪个队列,生产者无法决定。

交换机把消息发送给绑定过的所有队列 队列的消费者都能拿到消息。

实现一条消息被多个消费者消费

代码实现

生产者

public class MessageProvider3 {
    public static void main(String[] args) throws IOException, TimeoutException {
        Connection connection = RabbitMQUtils.getConnection();
        Channel channel = connection.createChannel();
        /**
         * 声明交换机
         * 参数1:交换机的名称
         * 参数2:交换机的类型  fanout广播类型的交换机(可以和任何队列进行绑定)
         */
        channel.exchangeDeclare("exchange_fanout","fanout");
        /**
         * 发布消息
         * 参数1:交换机的名称
         * 参数2;路由key 由于是广播类型的交换机,所以不用定义路由key
         * 参数3:对消息进行的额外设置
         * 参数4:消息主体
         */
        channel.basicPublish("exchange_fanout","",null,"hello rabbitmq".getBytes());
    }
}

 消费者

public class MessageConsumer3 {
    public static void main(String[] args) throws Exception {
        Connection connection = RabbitMQUtils.getConnection();
        Channel channel = connection.createChannel();
        //声明交换机
        channel.exchangeDeclare("exchange_fanout", "fanout");
        //声明队列
        channel.queueDeclare("q1", false, false, false, null);
        //将交换机和队列进行绑定 参数1: 队列的名称 参数2: 交换机的名称 参数3: 路由key
        channel.queueBind("q1", "exchange_fanout", "");
        //消费消息
        channel.basicConsume("q1", true, new DefaultConsumer(channel){
            @Override
            public void handleDelivery(String consumerTag, Envelope envelope, AMQP.BasicProperties properties, byte[] body) throws IOException {
                System.out.println(new String(body));
            }
        });
    }
}

一条消息会被所有消费者消费 

5.Direct

在前面的模型中,一条消息被所有的消费者消费,但是在某些特定的场景下,我们希望不同的消息被不同的队列消费

生产者向交换机发送消息时,会指定一个routing key

交换机在接收到消息时会根据这个key来对它绑定的队列的routing key进行匹配,匹配的才会发送过去。

代码实现

生产者

/**
 * 消息的生产者 -- direct消息模型
 */
public class MessageProvider {
    public static void main(String[] args) throws Exception {
        // 获取与RabbitMQ的连接
        Connection connection = RabbitMQUtils.getConnection();
        // 创建信道
        Channel channel = connection.createChannel();
        // 声明交换机,名称为exchange_direct,类型为direct
        channel.exchangeDeclare("exchange_direct", "direct");
        // 发布消息,发送到exchange_direct交换机,routingKey为add,消息内容为"hello, this is direct message"
        channel.basicPublish("exchange_direct", "add", null, "hello, this is direct message".getBytes());
        // 关闭信道
        channel.close();
        // 关闭连接
        connection.close();
    }
}

消费者1

 

/**
 * 消息的消费者1 -- direct类型
 */
public class ConsumerProvider_1 {
    public static void main(String[] args) throws Exception {
        // 获取与RabbitMQ的连接
        Connection connection = RabbitMQUtils.getConnection();
        // 创建信道
        Channel channel = connection.createChannel();
        // 声明交换机,名称为exchange_direct,类型为direct
        channel.exchangeDeclare("exchange_direct", "direct");
        // 声明队列,名称为queue_direct
        channel.queueDeclare("queue_direct", false, false, false, null);
        // 使用routingKey绑定队列和交换机,分别绑定add、update的routingKey
        channel.queueBind("queue_direct", "exchange_direct", "add");
        channel.queueBind("queue_direct", "exchange_direct", "update");
        // 消费消息,设置为自动确认
        channel.basicConsume("queue_direct", true, new DefaultConsumer(channel){
            @Override
            public void handleDelivery(String consumerTag, Envelope envelope, AMQP.BasicProperties properties, byte[] body) throws IOException {
                // 打印接收到的消息内容
                System.out.println(new String(body));
            }
        });
    }
}

消费者2

 

/**
 * 消息的消费者2 -- direct类型
 */
public class ConsumerProvider_2 {
    public static void main(String[] args) throws Exception {
        // 获取与RabbitMQ的连接
        Connection connection = RabbitMQUtils.getConnection();
        // 创建信道
        Channel channel = connection.createChannel();
        // 声明交换机,名称为exchange_direct,类型为direct
        channel.exchangeDeclare("exchange_direct", "direct");
        // 声明队列,名称为queue_direct1
        channel.queueDeclare("queue_direct1", false, false, false, null);
        // 使用routingKey绑定队列和交换机,分别绑定update、delete的routingKey
        channel.queueBind("queue_direct1", "exchange_direct", "update");
        channel.queueBind("queue_direct1", "exchange_direct", "delete");
        // 消费消息,设置为自动确认
        channel.basicConsume("queue_direct1", true, new DefaultConsumer(channel){
            @Override
            public void handleDelivery(String consumerTag, Envelope envelope, AMQP.BasicProperties properties, byte[] body) throws IOException {
                // 打印接收到的消息内容
                System.out.println(new String(body));
            }
        });
    }
}

消费者1的队列与交换机绑定的routing key分别是 add、update

消费者2的队列与交换机绑定的routing key分别是update、delete

示例代码中的生产者生产的消息的routing key是add,那么只有消费者1能接收到消息。

6.Topics

这个模型与Direct的原理是一样的,差异就在于它的routing key是有模糊匹配的机制。

但是它的routing key一般是由多个单词拼在一起,中间由”.“进行隔开,例如item.insert

这里的通配符有两种

分别是*和#

*可以匹配一个单词

#可以匹配一个或多个单词

代码演示

生产者

public class MessageProvider5 {
    public static void main(String[] args) throws Exception {
        Connection connection = RabbitMQUtils.getConnection();
        Channel channel = connection.createChannel();
        //声明topic类型的交换机
        channel.exchangeDeclare("topic_exchange", "topic");
        //发布消息
        channel.basicPublish("topic_exchange",
                             "user.product.add",
                             null,
                             "hello,this is topic message".getBytes());
        //关闭资源
        channel.close();
        connection.close();
    }
}

消费者1 

public class MessageConsumer5_1 {
    public static void main(String[] args) throws Exception {
        Connection connection = RabbitMQUtils.getConnection();
        Channel channel = connection.createChannel();
        //声明交换机
        channel.exchangeDeclare("topic_exchange", "topic");
        //声明队列
        channel.queueDeclare("topic_q1", false, false, false, null);
        //将队列和交换机进行绑定
        channel.queueBind("topic_q1", "topic_exchange", "user.*");
        //消费消息
        channel.basicConsume("topic_q1", true, new DefaultConsumer(channel){
            @Override
            public void handleDelivery(String consumerTag, Envelope envelope, AMQP.BasicProperties properties, byte[] body) throws IOException {
                System.out.println(new String(body));
            }
        });
    }
}

消费者2

 

public class MessageConsumer5_2 {
    public static void main(String[] args) throws Exception {
        Connection connection = RabbitMQUtils.getConnection();
        Channel channel = connection.createChannel();
        //声明交换机
        channel.exchangeDeclare("topic_exchange", "topic");
        //声明队列
        channel.queueDeclare("topic_q2", false, false, false, null);
        //将队列和交换机进行绑定
        channel.queueBind("topic_q2", "topic_exchange", "user.#");
        //消费消息
        channel.basicConsume("topic_q2", true, new DefaultConsumer(channel){
            @Override
            public void handleDelivery(String consumerTag, Envelope envelope, AMQP.BasicProperties properties, byte[] body) throws IOException {
                System.out.println(new String(body));
            }
        });
    }
}

7.RPC(了解)

  1. 客户端
    • 与 RabbitMQ 建立连接并创建信道。
    • 声明一个用于接收响应的回调队列。
    • 生成一个唯一的 correlationId,用于匹配请求和响应。
    • 发送请求消息到 RPC 队列,同时设置 replyTo 为回调队列名称,correlationId 为生成的唯一标识。
    • 从回调队列接收响应消息,并根据 correlationId 进行匹配。
  2. 服务端
    • 与 RabbitMQ 建立连接并创建信道。
    • 声明 RPC 队列。
    • 从 RPC 队列接收请求消息。
    • 处理请求消息,生成响应结果。
    • 将响应消息发送到 replyTo 指定的回调队列,并带上相同的 correlationId

 

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2331212.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

StarRocks 助力首汽约车精细化运营

作者&#xff1a;任智红&#xff0c;首汽约车大数据负责人 更多交流&#xff0c;联系我们&#xff1a;https://wx.focussend.com/weComLink/mobileQrCodeLink/334%201%202/ffbe5 导读&#xff1a; 本文整理自首汽约车大数据负责人任智红在 StarRocks 年度峰会上的演讲&#xf…

痉挛性斜颈康复助力:饮食调养指南

痉挛性斜颈患者除了积极治疗&#xff0c;合理饮食也能辅助缓解症状&#xff0c;提升生活质量。其健康饮食可从以下方面着手&#xff1a; 高蛋白质食物助力肌肉修复 痉挛性斜颈会导致颈部肌肉异常收缩&#xff0c;消耗较多能量&#xff0c;蛋白质有助于肌肉的修复与维持。日常可…

mysql镜像创建docker容器,及其可能遇到的问题

前提&#xff0c;已经弄好基本的docker服务了。 一、基本流程 1、目录准备 我自己的资料喜欢放在 /data 目录下&#xff0c;所以老规矩&#xff1a; 先进入 /data 目录&#xff1a; cd /data 创建 mysql 目录并进入&#xff1a; mkdir mysql cd mysql 2、镜像查找 docke…

JavaEE——线程的状态

目录 前言1. NEW2. TERMINATED3. RUNNABLE4. 三种阻塞状态总结 前言 本篇文章来讲解线程的几种状态。在Java中&#xff0c;线程的状态是一个枚举类型&#xff0c;Thread.State。其中一共分为了六个状态。分别为&#xff1a;NEW,RUNNABLE,BLOCKED,WAITING,TIMED_WAITING, TERMI…

RuntimeError: Error(s) in loading state_dict for ChartParser

一 bug错误 最近使用千问大模型有一个bug&#xff0c;报错信息如下 raise RuntimeError(Error(s) in loading state_dict for {}:\n\t{}.format( RuntimeError: Error(s) in loading state_dict for ChartParser:Unexpected key(s) in state_dict: "pretrained_model.em…

2025 年安徽交安安全员考试:利用记忆宫殿强化记忆​

安徽考生在面对交安安全员考试繁杂的知识点时&#xff0c;记忆宫殿是强大的记忆工具。选择一个熟悉且空间结构清晰的场所作为记忆宫殿&#xff0c;如自己居住的房屋。将房屋的不同区域&#xff0c;如客厅、卧室、厨房等&#xff0c;分别对应不同知识板块&#xff0c;像客厅对应…

安全编码课程 实验6 整数安全

实验项目 实现安全计数器&#xff1a;实现 Counter 结构&#xff0c;确保计数范围为 0~100。 实验要求&#xff1a; 1、使用 struct 封装计数值value&#xff1b; 2、计数器初值为 0&#xff1b; 3、increment() 方法增加计数&#xff0c;但不能超过 100&#xff1b; 4、decrem…

解决上传PDF、视频、音频等格式文件到FTP站点时报错“将文件复制到FTP服务器时发生错误。请检查是否有权限将文件放到该服务器上”问题

一、问题描述 可以将文本文件(.txt格式),图像文件(.jpg、.png等格式)上传到我们的FTP服务器上;但是上传一些PDF文件、视频等文件时就会报错“ 将文件复制到FTP服务器时发生错误。请检查是否有权限将文件放到该服务器上。 详细信息: 200 Type set to l. 227 Entering Pas…

【Linux操作系统】:信号

Linux操作系统下的信号 一、引言 首先我们可以简单理解一下信号的概念&#xff0c;信号&#xff0c;顾名思义&#xff0c;就是我们操作系统发送给进程的消息。举个简单的例子&#xff0c;我们在写C/C程序的时候&#xff0c;当执行a / 0类似的操作的时候&#xff0c;程序直接就挂…

经典频域分析法(Bode图、Nyquist判据) —— 理论、案例与交互式 GUI 实现

目录 经典频域分析法(Bode图、Nyquist判据) —— 理论、案例与交互式 GUI 实现一、引言二、经典频域分析方法的基本原理2.1 Bode 图分析2.2 Nyquist 判据三、数学建模与公式推导3.1 一阶系统的频域响应3.2 多极系统的 Bode 图绘制3.3 Nyquist 判据的数学描述四、经典频域分析…

使用scoop一键下载jdk和实现版本切换

安装 在 PowerShell 中输入下面内容&#xff0c;保证允许本地脚本的执行&#xff1a; set-executionpolicy remotesigned -scope currentuser然后执行下面的命令安装 Scoop&#xff1a; iwr -useb get.scoop.sh | iex国内用户可以使用镜像源安装&#xff1a;powershell iwr -us…

对状态模式的理解

对状态模式的理解 一、场景二、不采用状态模式1、代码2、缺点 三、采用状态模式1、代码1.1 状态类1.2 上下文&#xff08;这里指&#xff1a;媒体播放器&#xff09;1.3 客户端 2、优点 一、场景 同一个东西&#xff08;例如&#xff1a;媒体播放器&#xff09;&#xff0c;有一…

vue2(webpack)集成electron和 electron 打包

前言 之前发过一篇vue集成electron的文章&#xff0c;但是用vue3vite实现的&#xff0c;在vue2webpack工程可能不适用&#xff0c;所以这篇文章就主要介绍vue2webpack集成electron方法 创建项目 vue create vue-electron-demo目录架构 vue-electron-demo/ ├── src/ …

C++内存管理优化实战:提升应用性能与效率

&#x1f9d1; 博主简介&#xff1a;CSDN博客专家、CSDN平台优质创作者&#xff0c;高级开发工程师&#xff0c;数学专业&#xff0c;拥有高级工程师证书&#xff1b;擅长C/C、C#等开发语言&#xff0c;熟悉Java常用开发技术&#xff0c;能熟练应用常用数据库SQL server,Oracle…

redis数据迁移之通过redis-dump镜像

这里写目录标题 一、redis-dump 镜像打包1.1 安装windows docker1.2 idea项目创建1.3 idea镜像打包 二、redis数据迁移2.1 数据导出2.2 数据导入 一、redis-dump 镜像打包 没有找到可用的redis-dump镜像&#xff0c;需要自己打包一下&#xff0c;这里我是在idea直接打包的 1.…

redis导入成功,缺不显示数据

SpringBootTest class SecurityApplicationTests {AutowiredStringRedisTemplate template; //添加这句代码&#xff0c;自动装载&#xff0c;即可解决文章三处代码报错Testvoid contextLoads() {String compact Jwts.builder().signWith(Jwts.SIG.HS512.key().build()).subj…

从表格到序列:Swift 如何优雅地解 LeetCode 251 展开二维向量

文章目录 摘要描述题解答案题解代码分析示例测试及结果时间复杂度空间复杂度总结 摘要 在这篇文章中&#xff0c;我们将深入探讨 LeetCode 第 251 题——“展开二维向量”的问题。通过 Swift 语言&#xff0c;我们不仅会提供可运行的示例代码&#xff0c;还会结合实际场景进行…

小型园区网实验

划分VLAN SW3 [sw3]vlan batch 2 3 20 30 [sw3]interface GigabitEthernet 0/0/1 [sw3-GigabitEthernet0/0/1]port link-type access [sw3-GigabitEthernet0/0/1]port default vlan 2 [sw3-GigabitEthernet0/0/1]int g0/0/2 [sw3-GigabitEthernet0/0/2]port link-type acces…

c# 数据结构 链表篇 有关单链表的一切

本人能力有限,本文仅作学习交流与参考,如有不足还请斧正 目录 0.单链表好处 0.5.单链表分类 1.无虚拟头节点情况 图示: 代码: 头插/尾插 删除 搜索 遍历全部 测试代码: 全部代码 2.有尾指针情况 尾插 全部代码 3.有虚拟头节点情况 全部代码 4.循环单链表 几个…

VS Code连接服务器编写Python文件

1、下载 Visual Studio Code 2、打开扩展&#xff08;ctrl shift x ) 3、搜索 Remote - SSH&#xff0c;安装 4、F1 或者 点金左下角 5、选择&#xff1a;Remote-SSH: Connect to Host……&#xff0c;回车 6、第一次用的时候&#xff0c;VS Code 会提示添加 SSH 主机。输…