IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> Java知识库 -> RabbitMQ的消息应答以及不公平分发 -> 正文阅读

[Java知识库]RabbitMQ的消息应答以及不公平分发

RabbitMQ的消息应答以及不公平分发

RabbitMQ的消息应答机制

什么是RabbitMQ的消息应答

为了保证消息在发送过程中不丢失,rabbitmq 引入消息应答机制,消息应答就是:消费者在接收到消息并且处理该消息之后,告诉 rabbitmq 它已经处理了,rabbitmq 可以把该消息删除了

RabbitMQ的自动应答机制

在消息发送后就被认为已经传送成功,这种模式需要在高吞吐量和数据传输安全性方面做权衡,以这种模式仅适用在消费者可以高效并以某种速率能够处理这些消息的情况下使用。上一篇文章的简单程序就是自动确认

 /**
         * 消费者消费消息
         *  1. 消费哪个队列
         *  2. 消费成功之后是否要自动应答 true 代表自动 false 代表手动
         *  3. 推送的消息如何进行消费的接口回调
         *  4. 消费者取消消费的回调
         */
        channel.basicConsume(QUERE_NAME,false,deliverCallback,cancelCallback);

RabbitMQ的手动应答机制

手动应答的几种方法

  • Channel.basicAck 用于肯定确认,RabbitMQ 已知道该消息并且成功的处理消息,可以将其丢弃了
  • Channel.basicNack用于否定确认
  • Channel.basicReject 用于否定确认,与 Channel.basicNack 相比少一个参数,不处理该消息了直接拒绝,可以将其丢弃了

手动应答举例

  • 通过设置自动应答为false以及设置手动应答的确认方法,来开启手动应答
/**
 * 消息在手动应答时是不丢失的,放回队列中重新消费
 */
public class Worker01 {

    //设置队列名称
    public static final String TASK_QUEUE_NAME = "ack_queue";

    public static void main(String[] args) throws IOException, TimeoutException {
         //创建连接工厂
        ConnectionFactory factory = new ConnectionFactory();
        // 工厂IP,连接RabbitMQ队列
        factory.setHost("xxx.xxx.xxx.xxx");
        //用户名
        factory.setUsername("username");
        //密码
        factory.setPassword("password");

        Connection connection = factory.newConnection();
        Channel channel = connection.createChannel();
        
        // 推送的消息如何进行消费的接口回调
        DeliverCallback deliverCallback = (consumerTag, message) -> {
            System.out.println("接收到的消息:" + new String(message.getBody(),"UTF-8"));
            // 手动应答
            /**
             * 1. 消息的标记 tag
             * 2. 是否批量应答 false 不批量,true 批量
             */
        channel.basicAck(message.getEnvelope().getDeliveryTag(),false);
        };
        // 取消消息时的回调
        CancelCallback cancelCallback = consumerTag -> {
            System.out.println("消息消费被中断");
        };
      
        //采用手动应答
        Boolean autoAck = false;
        channel.basicConsume(TASK_QUEUE_NAME,autoAck,deliverCallback,cancelCallback);
    }
}
  • 上述channel.basicAck() 第二个参数批量不批量的解释:
    • true 代表批量应答 channel 上未应答的消息 比如说 channel 上有传送tag 的消息 5,6,7,8 当前 tag 是 8 那么此时5-8 的这些还未应答的消息都会被确认收到消息应答
    • false 同上面相比只会应答 tag=8 的消息 5,6,7 这三个消息依然不会被确认收到消息应答

RabbitMQ的不公平分发

上一篇文章中介绍了RabbitMQ的轮训分发,但是这种机制在某种策略下显得很不公平,比如当有多个消费者时,有一些消费者处理消息很快,有一些消费者处理消息很慢,这样处理快的消费者就在处理完的消息的时候一直处于空闲状态,而处理慢的那些消费者一直在干活,这种分配方式在这种情况下其实就不太好,所以此处介绍RabbitMQ的不公平分发。

代码示例

  • 通过设置 channel.basicQos(1); 来设置分发模式为不公平分发,下面我们对上篇的代码进行改造,让接收到消息进行不同时间的沉睡(此处应先关闭自动应答channel.basicConsume(QUERE_NAME,false,deliverCallback,cancelCallback);,通过手动应答时间的长短来模拟不同的处理消息的速度,以下为示例代码,work02与01仅仅是沉睡的时间不同,此处只写出work01
  • 生产者代码work01
/**
 * 消息在手动应答时是不丢失的,放回队列中重新消费
 */
public class Worker01 {

    //设置队列名称
    public static final String TASK_QUEUE_NAME = "ack_queue";

    public static void main(String[] args) throws IOException, TimeoutException {
        =========与上述相同生成信道===========
        System.out.println("C1等待接收消息处理,时间较短。。");

        // 推送的消息如何进行消费的接口回调
        DeliverCallback deliverCallback = (consumerTag, message) -> {
            // 沉睡1S
               try {
                Thread.sleep(1000 * 1);
            } catch (InterruptedException _ignored) {
                Thread.currentThread().interrupt();
            }
            System.out.println("接收到的消息:" + new String(message.getBody(),"UTF-8"));
            // 手动应答
            /**
             * 1. 消息的标记 tag
             * 2. 是否批量应答 false 不批量,true 批量
             */
            channel.basicAck(message.getEnvelope().getDeliveryTag(),false);
        };
        // 取消消息时的回调
        CancelCallback cancelCallback = consumerTag -> {
            System.out.println("消息消费被中断");
        };
        // 设置不公平分发
        channel.basicQos(1);
        //采用手动应答
        Boolean autoAck = false;
        channel.basicConsume(TASK_QUEUE_NAME,autoAck,deliverCallback,cancelCallback);
    }
}

  • 生产者代码,生产者也需要设置 channel.confirmSelect();来开启发布确认
/**
 * 消息在手动应答时是不丢失的,放回队列中重新消费
 */
public class Task02 {
    //设置队列名称
    public static final String TASK_QUEUE_NAME = "ack_queue";

    public static void main(String[] args) throws IOException, TimeoutException {
      
       ======与上述相同的生成信道=========
        // 开启发布确认 
        channel.confirmSelect();

        /**
         * 生成一个队列
         *  1. 队列名称
         *  2. 队列中的消息是否需要持久话(磁盘),默认情况消息存储在内存中
         *  3. 该队列是否只供一个消费者进行消费,是否进行消息共享,true可以多个消费者消费,false只能一个消费者消费
         *  4. 是否自动删除,最后一个消费者断开连接后,该队列是否自动删除 true自动删除,false则相反
         *  5. 其他参数
         */
        
      channel.queueDeclare(TASK_QUEUE_NAME,false,false,false,null);

        Scanner scanner = new Scanner(System.in);
        while(scanner.hasNext()){
            String message = scanner.next();
            // 设置生产者发送消息为持久化消息(要求保存在磁盘上),
            channel.basicPublish("",TASK_QUEUE_NAME, null,message.getBytes("UTF-8"));
            System.out.println("生产者发出消息:"+message);
        }
    }
}

实验结果

在这里插入图片描述
在这里插入图片描述

RabbitMQ的持久化

  • 队列的持久化
    RabbitMQ如果重启的话,之前创建的队列都会消失,如果要队列实现持久化,需要在声明队列的时候就把durable参数设置为持久化。
/**
         * 生成一个队列
         *  1. 队列名称
         *  2. 队列中的消息是否需要持久话(磁盘),默认情况消息存储在内存中
         *  3. 该队列是否只供一个消费者进行消费,是否进行消息共享,true可以多个消费者消费,false只能一个消费者消费
         *  4. 是否自动删除,最后一个消费者断开连接后,该队列是否自动删除 true自动删除,false则相反
         *  5. 其他参数
         */
        Boolean durable = true ; // 需要队列持久化
        channel.queueDeclare(TASK_QUEUE_NAME,durable,false,false,null);

  • 消息的持久化
    在推送消息的生产者上添加MessageProperties.PERSISTENT_TEXT_PLAIN这个属性,来使消息保存到磁盘,不写默认保存到内存
  // 设置生产者发送消息为持久化消息(要求保存在磁盘上),
 channel.basicPublish("",TASK_QUEUE_NAME, MessageProperties.PERSISTENT_TEXT_PLAIN,message.getBytes("UTF-8"));
         

尚硅谷B站RabbitMQ教程:尚硅谷RabbitMQ

  Java知识库 最新文章
计算距离春节还有多长时间
系统开发系列 之WebService(spring框架+ma
springBoot+Cache(自定义有效时间配置)
SpringBoot整合mybatis实现增删改查、分页查
spring教程
SpringBoot+Vue实现美食交流网站的设计与实
虚拟机内存结构以及虚拟机中销毁和新建对象
SpringMVC---原理
小李同学: Java如何按多个字段分组
打印票据--java
上一篇文章      下一篇文章      查看所有文章
加:2022-02-28 15:14:49  更:2022-02-28 15:15:10 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2024年11日历 -2024/11/24 11:40:59-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码