2024年大数据最新RabbitMQ的工作队列有哪些?带你玩转工作队列_mq指定工作队列,2024年最新大数据开发高级工程师系列学习路线介绍

img
img
img

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,涵盖了95%以上大数据知识点,真正体系化!

由于文件比较多,这里只是将部分目录截图出来,全套包含大厂面经、学习笔记、源码讲义、实战项目、大纲路线、讲解视频,并且后续会持续更新

需要这份系统化资料的朋友,可以戳这里获取

🎁作者简介:在校大学生一枚,Java领域新星创作者,Java、Python正在学习中,期待和大家一起学习一起进步~
💗个人主页:我是一棵卷心菜的个人主页
🔶本文专栏:RabbitMQ学习
📕自我提醒:多学多练多思考,编程能力才能节节高!

文章目录

一、准备工作

  • 为了方便后续代码的简洁性以及可读性,我们先准备一个工具类,代码如下:
public class RabbitMqUtils {
    //创建连接的方法
    public static Connection getConnection() throws IOException, TimeoutException {
        //创建一个连接工厂
        ConnectionFactory factory = new ConnectionFactory();
        //设置一些参数
        factory.setHost("192.168.205.131");
        factory.setUsername("cabbage");
        factory.setPassword("cabbage");
        factory.setPort(5672);
        return factory.newConnection();
    }
    //得到一个连接的 channel
    public static Channel getChannel() throws Exception {
        return getConnection().createChannel();
    }
    //关闭channel
    public static void closeChannel() throws Exception {
        getChannel().close();
    }
    //关闭连接
    public static void closeConnection() throws IOException, TimeoutException {
        getConnection().close();
    }
}


二、channel方法讲解

  • 在学习工作队列的过程中,我们还会用到很多方法以及方法参数的使用,很有必要弄清楚!

1、queueDeclare

  • 源码:
Queue.DeclareOk queueDeclare(String queue, boolean durable, boolean exclusive, boolean autoDelete,
                                 Map<String, Object> arguments) throws IOException;

  • 参数讲解:
参数含义注意点
queue队列名称
durable队列中的消息是否持久化队列的数据默认是存放到内存中的,但rabbitmq重启会丢失队列,如果想重启之后数据还存在就要使队列持久化 ( 设置durable为true ),保存到Erlang自带的数据库中,当rabbitmq重启之后会读取该数据库。
autoDelete当最后一个消费者断开连接之后队列是否自动被删除如果生产者声明了一个queue,此队列从来没有消费者连接过,那么 即使consumers = 0,队列也不会自动删除
arguments其它参数
exclusive该队列是否只供一个消费者进行消费 or 当connection关闭时是否删除队列如果队列是排外的,会对当前队列加锁,exclusive等于true的话用于一个队列只能有一个消费者来消费的场景

注意:

rabbitmq消息持久化有两个步骤:一是队列持久化;二是消息持久化。当durable=true时,如果只设置该值,是把队列进行持久化(重启后消息还是会丢失);所以我们还应该在发送消息时对消息进行持久化。代码如下:

//发送消息
channel.basicPublish("交换机名称", "路由类型", MessageProperties.PERSISTENT\_TEXT\_PLAIN, "消息主体".getBytes());

MessageProperties.PERSISTENT_TEXT_PLAIN就是对消息进行持久化,但仍需注意的是,即使设置了这两种操作,消息也不是100%会被持久化的!

2、basicPublish

  • 源码:
void basicPublish(String exchange, String routingKey, BasicProperties props, byte[] body) throws IOException;

  • 参数讲解:
参数含义
exchange要将消息发送到交换机的名称
routingKey路由key
props一些配置信息
body消息内容

3、basicConsume

  • 源码:
String basicConsume(String queue, boolean autoAck, DeliverCallback deliverCallback, CancelCallback cancelCallback) throws IOException;

  • 参数讲解:
参数含义
queue消费队列的名称
autoAck消费成功之后是否要自动应答
deliverCallback当一个消息发送过来后的回调接口
cancelCallback当一个消费者取消订阅时的回调接口

注意:

自动确认 autoAck = true 只要消息从队列中获取,无论消费者获取到消息后是否成功消费消息,都认为是消息已经成功消费。
手动确认 autoAck = false 消费者从队列中获取消息后,服务器会将该消息标记为不可用状态,等待消费者的反馈,如果消费者一直没有反馈,那么该消息将一直处于不可用状态。


三、简单模式

在这里插入图片描述

  • 生产者代码:我们向消费者发送了消息你好,MessageQueue
public class Producer {
    public static void main(String[] args) throws Exception {
        Connection connection = RabbitMqUtils.getConnection();
        Channel channel = RabbitMqUtils.getChannel();
        channel.queueDeclare("hello", false, false, false, null);
        channel.basicPublish("", "hello", null, "你好,MessageQueue".getBytes());
        // 释放资源
        channel.close();
        connection.close();
    }
}

当我们执行生产者代码,就会产生一个带有数据的队列:

在这里插入图片描述

  • 消费者代码:
public class Consumer {
    public static void main(String[] args) throws Exception {
        Connection connection = RabbitMqUtils.getConnection();
        Channel channel = RabbitMqUtils.getChannel();
        // 声明 接收消息
        DeliverCallback deliverCallback = (consumerTag, message) -> {
            System.out.println(new String(message.getBody()));
        };
        // 取消消息时的回调
        CancelCallback cancelCallback = consumerTag -> {
            System.out.println("消息消费被中断");
        };
        channel.basicConsume("hello", true, deliverCallback, cancelCallback);
    }
}

当我们执行完消费者代码,可以看到控制台界面和消息被消费

在这里插入图片描述

在这里插入图片描述


四、Work模式

在这里插入图片描述

一个生产者对应多个消费者,但是一条消息只能由一个消费者消费。
主要有两种模式:一是轮询模式的分发,一个消费者一条消息,按均分配;二是公平分发,根据消费者的消费能力进行公平分发,能力越高,消费能力越强,按劳分配。

  • 接下来主要介绍轮询模式,生产者代码:
public class Producer {
    public static void main(String[] args) throws Exception {
        Channel channel = RabbitMqUtils.getChannel();
        channel.queueDeclare("work", false, false, false, null);
        for (int i = 1; i <= 10; i++) {
            Thread.sleep(i \* 10);
            channel.basicPublish("", "work", null, (i + "I'am rabbitMQ").getBytes());
        }
        RabbitMqUtils.closeChannel();
        RabbitMqUtils.closeConnection();
    }
}

  • 消费者1的代码:
public class Work\_1 {
    public static void main(String[] args) throws Exception {
        Channel channel = RabbitMqUtils.getChannel();
        DeliverCallback deliverCallback = (consumerTag, message) -> {
            try {
                Thread.sleep(1000);
                System.out.println(new String(message.getBody()));
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            channel.basicAck(message.getEnvelope().getDeliveryTag(), false);
        };
        CancelCallback cancelCallback = (consumerTag) -> {
            System.out.println("消息消费被中断");
        };
// channel.basicQos(1);
        channel.basicConsume("work", false, deliverCallback, cancelCallback);
        RabbitMqUtils.closeChannel();
        RabbitMqUtils.closeConnection();
    }
}

  • 消费者2的代码:
public class Work\_2 {
    public static void main(String[] args) throws Exception {
        Channel channel = RabbitMqUtils.getChannel();
        // 声明 接收消息
        DeliverCallback deliverCallback = (consumerTag, message) -> {
            try {
                Thread.sleep(5 \* 1000);
                System.out.println(new String(message.getBody()));
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            //返回确定状态,由于我们是设置手动返回状态的,如果没有调用该方法,那么消息仍然在我们队列中,不会被删除
            channel.basicAck(message.getEnvelope().getDeliveryTag(), false);
        };
        CancelCallback cancelCallback = consumerTag -> {
            System.out.println("消息消费被中断");
        };
// channel.basicQos(1);
        channel.basicConsume("work", false, deliverCallback, cancelCallback);
        RabbitMqUtils.closeChannel();
        RabbitMqUtils.closeConnection();
    }
}

  • 接着开始测试代码,首先让生产者发送10条消息到队列中,然后运行消费者代码,打开控制台,如图:

在这里插入图片描述

在这里插入图片描述

我们可以发现,即使两个消费者的睡眠时间不同(处理消息的能力不一样),但是消费消息的数目确实一样的。

分析运行结果:

1、消费者1和消费者2获取到的消息内容是不同的,即同一个消息只能被一个消费者获取。
2、两个消费者获取消息的条数是一样的。

  • 而对于公平分发,我们只需要在上述代码中添加下列代码:
channel.basicQos(1);


五、发布/订阅模式

在这里插入图片描述

P:在订阅模式中,多了一个Exchange角色,生产者不再发送到队列中,而是发送给交换机。
Exchange:一方面接收生产者发送的消息,另一方面知道如何处理消息;而到底如何处理,取决于Exchange的类型。有常见的以下三种类型:

  • Fanout:广播,将消息交给所有绑定到交换机的队列
  • Direct:定向,把消息交给符合指定routing key的队列
  • Topic:通配符,把消息交给符合routing pattern的队列

注意:交换机只负责转发消息,不具备存储信息的能力,因此如果没有任何队列与交换机绑定,或者没有符合路由规则的队列,那么信息会丢失!

  • 生产者代码:
public class Producer {
    public static void main(String[] args) throws Exception {
        Channel channel = RabbitMqUtils.getChannel();
        /\*
 String exchange:交换机名称
 BuiltinExchangeType type:交换机类型
 boolean durable:是否持久化
 boolean autoDelete:自否自动删除
 boolean internal:内部使用,一般为false


![img](https://img-blog.csdnimg.cn/img_convert/f2d0d2b0ffc63d3bfcd7d168687f42cb.png)
![img](https://img-blog.csdnimg.cn/img_convert/31ce22a4123d1ecd40ef533bcc219e9f.png)
![img](https://img-blog.csdnimg.cn/img_convert/955d1d49807f1d89a30fc8dbfbd641ea.png)

**既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,涵盖了95%以上大数据知识点,真正体系化!**

**由于文件比较多,这里只是将部分目录截图出来,全套包含大厂面经、学习笔记、源码讲义、实战项目、大纲路线、讲解视频,并且后续会持续更新**

**[需要这份系统化资料的朋友,可以戳这里获取](https://bbs.csdn.net/forums/4f45ff00ff254613a03fab5e56a57acb)**

pe type:交换机类型
 boolean durable:是否持久化
 boolean autoDelete:自否自动删除
 boolean internal:内部使用,一般为false


[外链图片转存中...(img-7vTLVGYs-1715261669110)]
[外链图片转存中...(img-NriQ1omk-1715261669110)]
[外链图片转存中...(img-SQ6ntwv6-1715261669110)]

**既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,涵盖了95%以上大数据知识点,真正体系化!**

**由于文件比较多,这里只是将部分目录截图出来,全套包含大厂面经、学习笔记、源码讲义、实战项目、大纲路线、讲解视频,并且后续会持续更新**

**[需要这份系统化资料的朋友,可以戳这里获取](https://bbs.csdn.net/forums/4f45ff00ff254613a03fab5e56a57acb)**

  • 4
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值