RabbitMQ--消费端单线程与多线程

🌟 一、基础概念

模型 消费者数量 每个消费者内部线程数 顺序性 场景说明
单消费者单线程 1 1 ✅ 保序 处理逻辑简单,保证顺序的常见场景
单消费者多线程 1 >1 ❌ 不保序 提升处理能力,放弃顺序要求
多消费者单线程 >1 1 ❌ 不保序 多个队列/分区消费,提升并发
多消费者多线程 >1 >1 ❌ 不保序 高并发场景下批量处理,放弃顺序

java 复制代码
        concurrency# 初始消费者线程数
        max-concurrency# 最大消费者线程数
        prefetch# 每个消费者预取的消息数
  1. concurrency: 2

    • 表示初始创建的消费者线程数量

    • 系统启动时会立即创建 2 个消费者线程

    • 这些线程会持续监听消息队列

  2. max-concurrency: 2

    • 表示允许的最大消费者线程数量

    • 这里设置为 2(与 concurrency 相同),表示线程数不会动态扩展

    • 如果设置 max-concurrency > concurrency,系统会在负载高时动态增加消费者

详细解释:

concurrency和max-concurrency 不会影响每个消费者是否是多线程执行,只会导致有多个消费者线程,只有用线程池才会导致每个消费者多线程消费

而没有用线程池,也设置prefetch是因为****消息被大量预取,单线程处理不过来时堆积等待,单线程并不会影响消息的顺序性,只有使用了线程池才会影响

使用了线程池一定会导致消息顺序性问题这与设不设置prefetch无关,因为使用线程池后,任务交个线程池就返回了属于异步

举个例子:

  1. RabbitMQ 给消费者推送消息1,消费者收到,提交给线程池任务A(耗时长)。

  2. 消费者马上ACK消息1(因为业务交给线程池了,自己处理完毕的感觉

  3. RabbitMQ 再给消费者推送消息2,消费者收到,提交给线程池任务B(耗时短)。

  4. R线程池调度先跑完任务B,后跑任务A。

✅ 单消费者 + 单线程消费

  • 保证顺序:消费者内部串行执行。

  • 配置关键

    java 复制代码
    spring:
      rabbitmq:
        listener:
          simple:
            concurrency: 1
            max-concurrency: 1
            prefetch: 1
  • 消费者代码

    java 复制代码
    @Component
    public class MultiConsumerSingleThread {
    
        @RabbitListener(queues = "order_queue", concurrency = "2")
        public void receive(String message) {
            System.out.println("🧾 [线程:" + Thread.currentThread().getName() + "] 收到消息:" + message);
            try {
                Thread.sleep(500);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
    }

❌ 单消费者 + 多线程消费

  • 不保顺序:一个消费者使用线程池异步处理消息。

  • 配置关键:默认配置 + 手动异步处理

  • 消费者代码

    java 复制代码
    @Component
    public class MultiThreadConsumer {
    
        private final ExecutorService executor = Executors.newFixedThreadPool(5);
    
        @RabbitListener(queues = "order_queue")
        public void receive(String message) {
            executor.submit(() -> {
                System.out.println("🧾 [线程:" + Thread.currentThread().getName() + "] 收到消息:" + message);
                try {
                    Thread.sleep(500); // 模拟耗时
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            });
        }
    }
  • 说明:消息提交到线程池,先到的不一定先处理完成,顺序可能乱。


❌ 多消费者 + 单线程消费

  • 不保顺序:多个消费者实例轮询分配消息,各自顺序保留,但整体顺序错乱。

  • 配置关键

    java 复制代码
    spring:
      rabbitmq:
        listener:
          simple:
            concurrency: 2
            max-concurrency: 2
            prefetch: 1
  • 消费者代码(共享类,也可拆成多个类模拟多实例)

    java 复制代码
    @Component
    public class MultiConsumerSingleThread {
        //concurrency = "2":它和配置文件中的 concurrency: 2 作用一致,但优先级更高。
        @RabbitListener(queues = "order_queue", concurrency = "2")
        public void receive(String message) {
            System.out.println("🧾 [线程:" + Thread.currentThread().getName() + "] 收到消息:" + message);
            try {
                Thread.sleep(500);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
    }

❌ 多消费者 + 多线程消费

  • 不保顺序:每个消费者又使用线程池异步处理消息,最大吞吐量模式。

  • 适合场景:数据导入、日志收集、发送通知等对顺序无要求的批量处理。

  • 配置关键

    java 复制代码
    spring:
      rabbitmq:
        listener:
          simple:
            concurrency: 3
            max-concurrency: 3
            prefetch: 10
  • 消费者代码

    java 复制代码
    @Component
    public class MultiConsumerMultiThread {
    
        private final ExecutorService executor = Executors.newFixedThreadPool(10);
    
        @RabbitListener(queues = "order_queue", concurrency = "3")
        public void receive(String message) {
            executor.submit(() -> {
                System.out.println("🧾 [线程:" + Thread.currentThread().getName() + "] 收到消息:" + message);
                try {
                    Thread.sleep(500);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            });
        }
    }

🧠 补充说明

  • concurrency: 控制并发消费者数量,等于消费者数。

  • prefetch: 控制每个消费者本地最多拉取多少条消息(如 1 表示严格串行处理)。

  • 每个 @RabbitListener 本质上是一个容器,可以通过 concurrency 配置"实例个数"。

相关推荐
Hugh-Yu-1301236 分钟前
二元一次方程组求解器c++代码
开发语言·c++·算法
weixin_5206498711 分钟前
C#进阶-特性全知识点总结
开发语言·c#
文祐13 分钟前
C++类之虚函数表及其内存布局
开发语言·c++
亦暖筑序26 分钟前
Spring AI Alibaba 报错合集:我踩过的那些坑
java·后端
编程大师哥30 分钟前
C++类和对象
开发语言·c++·算法
M1582276905531 分钟前
工业 CAN 总线无线互联利器|4 路 CAN 转 4G/WiFi 网关 产品介绍
开发语言·php
石榴树下的七彩鱼35 分钟前
OCR 识别不准确怎么办?模糊 / 倾斜 / 反光图片优化实战(附完整解决方案 + 代码示例)
图像处理·人工智能·后端·ocr·api·文字识别·图片识别
indexsunny1 小时前
互联网大厂Java面试实战:核心技术与微服务架构在电商场景中的应用
java·spring boot·redis·kafka·maven·spring security·microservices
摇滚侠1 小时前
Java 多线程基础 Java Multithreading Basics
java
burning_maple1 小时前
AI 工程实战指南:从零开始构建 AI 应用
开发语言·人工智能