我的编程空间,编程开发者的网络收藏夹
学习永远不晚

JavaRabbitMQ的工作队列与消息应答详解

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

JavaRabbitMQ的工作队列与消息应答详解

Work Queues

工作队列(任务队列)主要思想是避免立即执行资源密集型任务,而不得不等待它完成,相反我们安排任务在之后执行。我们把任务封装为消息并将其发送到队列。在后台运行的工作进程将弹出任务并最终执行作业。当有多个工作线程时,这些工作线程将一起处理这些任务。

其实就是生产者发送大量的消息,发送到队列之后,由多个消费者(工作线程)来处理消息,并且每个消息只能被处理一次。

在这里插入图片描述

1. 轮询分发消息

多个工作线程按照次序每来一个消息执行一次。

1.1 抽取工具类

直接通过信息获取信道


public class RabbitMQUtils {
    public static Channel getChannel() throws Exception{
        ConnectionFactory factory = new ConnectionFactory();
        factory.setHost("1");
        factory.setUsername("guest");
        factory.setPassword("guest");
        Connection connection = factory.newConnection();
        return connection.createChannel();
    }
}

1.2 编写两个工作线程

Work2和Work1代码没有区别,只需要对它做出区分即可。

public class Worker1 {
    // 指定队列名称
    private static final String QUEUE_NAME = "work_queue";

    public static void main(String[] args) throws Exception {

        // 获取信道
        Channel channel = RabbitMQUtils.getChannel();

        // 声明:接收消息回调
        DeliverCallback deliverCallback = (consumerTag, message) -> {
            System.out.println("工作线程01:"+ new String(message.getBody()));
        };

        // 声明:取消消费回调
        CancelCallback cancelCallback = consumerTag -> {
            System.out.println("工作线程01取消接收:"+consumerTag);
        };

        System.out.println("工作线程01启动完成......");

        channel.basicConsume(QUEUE_NAME,false,deliverCallback,cancelCallback);
    }
}

1.3 编写生产者

public class Producer {

    private static final String QUEUE_NAME = "work_queue";

    public static void main(String[] args) throws Exception {

        Channel channel = RabbitMQUtils.getChannel();


        // 产生队列
        channel.queueDeclare(QUEUE_NAME,false,false,true,null);

        // 消息体
        Scanner scanner = new Scanner(System.in);
        int i = 1;
        while (scanner.hasNext()){
            String msg = scanner.next();
            msg = msg + i;
            channel.basicPublish("",QUEUE_NAME,null,msg.getBytes());
            System.out.println("发送成功:" + msg);
        }

        System.out.println("----------==========发送完毕==========----------");
    }

}

1.4 运行测试

先启动两个工作线程,再启动生产者。

出现404异常请参考下方1.6

生产者发送情况:

在这里插入图片描述

轮询状态下两个工作队列接收状态:

在这里插入图片描述

在这里插入图片描述

1.5 异常情况

在先启动两个消费者线程时,会提示404找不到队列

Caused by: com.rabbitmq.client.ShutdownSignalException: channel error; protocol method: #method<channel.close>(reply-code=404, reply-text=NOT_FOUND - no queue 'work_queue' in vhost '/', class-id=60, method-id=20)

发生这个情况的原因很显然是因为先启动了消费者,但是在RabbitMQ中没有创建相对应的队列名称,解决方法可以:

1.先启动生产者创建队列(也可以在RabbitMQ中创建队列);

2.再启动消费者就不会产生这个错误;

3.再在生产者中使用Scanner类去发送消息测试。

2. 消息应答

消费者在接收到消息并且处理该消息之后,告诉RabbitMQ它已经处理了,RabbitMQ可以删除消息。其目的就是为了保护消息在被处理之前不会消失。

2.1 自动应答

这种方式发送后就被认定为已经传送成功,所以在消息接收到之前消费者的连接或者channel关闭,那么这个消息就会丢失。其特点是消费者可以传递过载的消息,对传递的消息没有限制,但如果因内存耗尽消费者线程被系统杀死,就会使得多条消息丢失。所以这个模式需要在数据安全性和吞吐量之间选择,适合使用在消费者可以高效并以某种速率能够处理这些消息的情况下使用。

所以自动应答的方式局限性很高。

2.2 手动应答

优点:可以批量应答和减少网络拥挤。

1.channel.basicAck(long deliveryTag, boolean multiple);:肯应应答,处理完消息之后提醒RabbitMQ可以删除当前队列,deliveryTag:当前队列中选中的消息;multiple:是否批量应答。

2.channel.basicNack(long deliveryTag, boolean multiple, boolean requeue):否定应答,

3.channel.basicReject(long deliveryTag, boolean requeue):否定并且拒绝应答。

2.3 消息自动重新入队

如果消费者因为一些原因失去了对RabbitMQ的连接,导致没有发送ACK确认,RabbitMQ就会对该消息进行重新排队,并且分发给可以处理该消息的消费者,所以即使某个消费者死亡,也可以保证消息不会丢失。

2.4 手动应答测试

测试目的:在手动应答状态下不会发生消息丢失的情况。

测试方法:

1.创建两个消费者;

2.使用工具类使线程睡眠一定时间;

3.在睡眠时关闭线程,看能否自动重新入队。

2.4.1 生产者代码


public class Producer1 {

    // 指定队列名
    private static final String TASK_QUEUE_RES = "queue_res";

    public static void main(String[] args) throws Exception {
        Channel channel = RabbitMQUtils.getChannel();
        channel.queueDeclare(TASK_QUEUE_RES,false,false,false,null);
        Scanner scanner = new Scanner(System.in);
        int i = 0;
        while (scanner.hasNext()){
            i++;
            String msg = scanner.next() + i;
            channel.basicPublish("",TASK_QUEUE_RES,null,msg.getBytes(StandardCharsets.UTF_8));
            System.out.println("发送消息:'" + msg + "'成功");
        }
    }
}

2.4.2 消费者代码


public class Worker1 {

    private static final String TASK_QUEUE_RES = "queue_res";

    public static void main(String[] args)  throws Exception{
        Channel channel = RabbitMQUtils.getChannel();
        System.out.println("线程A等待接收......");

        DeliverCallback deliverCallback = (consumerTag, message) -> {
            // 模拟并发沉睡一秒
            try {
                Thread.sleep(1000);
                System.out.println("线程A接收消息:"+ new String(message.getBody(), StandardCharsets.UTF_8));
                
                channel.basicAck(message.getEnvelope().getDeliveryTag(),false);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        };
        channel.basicConsume(TASK_QUEUE_RES,false,deliverCallback,
                consumerTag -> {
                    System.out.println(consumerTag + "消费者取消消费");
                });

    }
}

Worker2类和1区别不大,将名称改成B再将睡眠事件改成30即可。

2.4.3 测试

测试方法:

1.先启动生产者创建队列;

2.启动两个消费者接收消息;

3.因为是轮询方式,所以A线程接收之后肯定是B线程接收,在睡眠时关闭B线程,如果A线程接收到说明测试成功。

发送消息:

在这里插入图片描述

线程A接收:

在这里插入图片描述

再发送消息:

在这里插入图片描述

关闭线程B线程A接收到消息:

在这里插入图片描述

测试成功!

总结

本篇文章就到这里了,希望能够给你带来帮助,也希望您能够多多关注编程网的更多内容!   

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

JavaRabbitMQ的工作队列与消息应答详解

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

AndroidMessageQueue消息队列主要作用详解

Android消息机制主要指的是Handler的运行机制及其所依赖的MessageQueue和Looper的工作过程,Handler、MessageQueue、Looper组成一个相互联系的整体。本文先从MessageQueue的源码来说明其实现原理
2023-02-16

Go WaitGroup与消息队列的优雅协作

Go WaitGroup与消息队列可以通过优雅协作来提高程序的性能和可维护性。Go WaitGroup是一种用于等待一组goroutine完成任务的机制。它可以确保在所有goroutine完成任务之前,主goroutine不会退出。使用Wa
2023-10-12

队列技术在PHP与MySQL中的消息分发和消息回调的应用

一、队列技术的概念和原理队列是一种先进先出(FIFO)的数据结构,用于存储和处理需要异步处理的任务。队列中的元素可以是任何类型的任务,如发送电子邮件、生成报表、处理用户请求等。队列技术的基本原理是将任务添加到队列中,并由一个或多个工作进程从
2023-10-21

队列技术在PHP与MySQL中的消息延迟和消息重试的应用

摘要:随着Web应用程序的不断发展,对于高并发处理和系统可靠性方面的需求越来越高。队列技术作为一种解决方案,被广泛应用于PHP与MySQL中,以实现消息延迟和消息重试的功能。本文将介绍队列技术在PHP与MySQL中的应用,包括队列的基本原理
2023-10-21

队列技术在PHP与MySQL中的消息拆分和消息合并的应用

引言:队列技术是一种非常重要的数据结构,它在分布式系统中起到了至关重要的作用。在PHP与MySQL中,队列技术可以被广泛应用于消息拆分和消息合并的场景中。本文将介绍队列技术在PHP与MySQL中的应用,并提供具体的代码示例。一、队列技术的概
2023-10-21

队列技术在PHP与MySQL中的消息过滤和消息订阅的应用

随着互联网的快速发展,消息处理已成为各种应用程序中至关重要的一环。在PHP与MySQL的数据库环境下,队列技术的应用可以满足消息的高效处理需求。本文将介绍队列技术在PHP与MySQL中消息过滤和消息订阅的应用,并给出具体的代码示例。一、消息
2023-10-21

队列技术在PHP与MySQL中的消息排序和消息合并的应用

随着互联网的快速发展,大量的数据和信息传输成为了一种常见的需求。为了处理这些大规模的数据和信息,队列技术应运而生。队列是一种先进先出(FIFO)的数据结构,它可以在多个系统之间传递消息,并确保消息按照顺序进行处理。在PHP与MySQL中,队
2023-10-21

详解redis是如何实现队列消息的ack

前言 由于公司提供的队列实在太过于蛋疼而且还限制不能使用其他队列,但为了保证数据安全性需要一个可以有ack功能的队列。 原生的redis中通过L/R PUSH/POP方式来实现队列的功能,这个当然是没办法满足需求的(没有ack功能),所以需
2022-06-04

队列技术在PHP与MySQL中的消息过滤和消息中间件的应用

引言:随着互联网的高速发展,我们的应用程序不再仅仅是简单的网页,而是涉及到大量的异步任务和消息传递。为了使我们的应用程序更加健壮和高效,队列技术成为了一个必不可少的工具。本文将介绍队列技术在PHP与MySQL中的消息过滤和消息中间件的应用,
2023-10-21

队列的消息持久化和消息验证在PHP与MySQL中的应用场景

引言队列是一个常用的应用程序组件,它可以用于在不同的系统之间传递消息。在复杂的系统架构中,消息队列可以起到解耦系统组件、提高系统可靠性和性能的作用。本文将介绍队列的消息持久化和消息验证在PHP与MySQL中的应用场景,并提供具体的代码示例。
2023-10-21

队列的消息去重和消息幂等性在PHP与MySQL中的应用场景

引言:随着互联网技术的不断发展,队列成为了各种分布式系统中重要的基础组件之一。而在队列的应用过程中,消息的去重和消息的幂等性是两个常见的问题,尤其在高并发场景下更为突出。本文将详细介绍队列的消息去重和消息幂等性在PHP与MySQL中的应用场
2023-10-21

队列技术在PHP与MySQL中的消息去重和消息幂等性的应用

摘要:随着互联网应用的不断发展,消息队列成为了处理高并发、异步操作的重要工具之一。在PHP与MySQL中,如何利用队列来解决消息去重和消息幂等性的问题呢?本文将介绍使用Redis和MySQL实现这两个功能的具体代码示例。引言消息队列是一种在
2023-10-21

队列的消息持久化和消息去重在PHP与MySQL中的应用场景

队列是一种常见的数据结构,在软件开发中被广泛应用于异步消息处理、任务调度、日志收集等场景。其中,消息持久化和消息去重是队列的两个重要特性,能够保证消息的可靠性和数据的一致性。在PHP和MySQL中,队列的应用可以通过Redis作为消息中间件
2023-10-21

队列技术在PHP与MySQL中的消息持久化和消息重播的应用

随着互联网的迅速发展,用户越来越追求高效、快速的体验,对于网站和应用程序来说,处理大量的并发请求是一项重要的挑战。为了解决这个问题,队列技术成为了开发人员的首选解决方案。本文将介绍如何在PHP与MySQL中使用队列技术实现消息持久化和消息重
2023-10-21

队列的消息排序和消息优先级在PHP与MySQL中的应用场景

队列(Queue)是计算机科学中常用的一种数据结构,它按照先进先出(First-In-First-Out,FIFO)的原则进行操作。队列广泛应用于消息传递、任务调度、事件驱动等场景中。而在实际应用中,有时我们需要对队列中的消息进行排序或者给
2023-10-21

详解Python操作RabbitMQ服务器消息队列的远程结果返回

先说一下笔者这里的测试环境:Ubuntu14.04 + Python 2.7.4 RabbitMQ服务器sudo apt-get install rabbitmq-serverPython使用RabbitMQ需要Pika库sudo pip
2022-06-04

C++实现一个简单消息队列的示例详解

消息队列在多线程的场景有时会用到,尤其是线程通信跨线程调用的时候,就可以使用消息队列进行通信。本文将利用C++实现一个简单的消息队列,感兴趣的可以了解一下
2022-12-15

编程热搜

  • Python 学习之路 - Python
    一、安装Python34Windows在Python官网(https://www.python.org/downloads/)下载安装包并安装。Python的默认安装路径是:C:\Python34配置环境变量:【右键计算机】--》【属性】-
    Python 学习之路 - Python
  • chatgpt的中文全称是什么
    chatgpt的中文全称是生成型预训练变换模型。ChatGPT是什么ChatGPT是美国人工智能研究实验室OpenAI开发的一种全新聊天机器人模型,它能够通过学习和理解人类的语言来进行对话,还能根据聊天的上下文进行互动,并协助人类完成一系列
    chatgpt的中文全称是什么
  • C/C++中extern函数使用详解
  • C/C++可变参数的使用
    可变参数的使用方法远远不止以下几种,不过在C,C++中使用可变参数时要小心,在使用printf()等函数时传入的参数个数一定不能比前面的格式化字符串中的’%’符号个数少,否则会产生访问越界,运气不好的话还会导致程序崩溃
    C/C++可变参数的使用
  • css样式文件该放在哪里
  • php中数组下标必须是连续的吗
  • Python 3 教程
    Python 3 教程 Python 的 3.0 版本,常被称为 Python 3000,或简称 Py3k。相对于 Python 的早期版本,这是一个较大的升级。为了不带入过多的累赘,Python 3.0 在设计的时候没有考虑向下兼容。 Python
    Python 3 教程
  • Python pip包管理
    一、前言    在Python中, 安装第三方模块是通过 setuptools 这个工具完成的。 Python有两个封装了 setuptools的包管理工具: easy_install  和  pip , 目前官方推荐使用 pip。    
    Python pip包管理
  • ubuntu如何重新编译内核
  • 改善Java代码之慎用java动态编译

目录