【多线程初阶】阻塞队列 生产者消费者模型

news2025/7/28 13:55:01

文章目录

  • 一、阻塞队列
  • 二、生产者消费者模型
    • (一)概念
    • (二)生产者消费者的两个重要优势(阻塞队列的运用)
      • 1) 解耦合(不一定是两个线程之间,也可以是两个服务器之间)
      • 2) 削峰填谷
    • (三)生产者消费者模型付出的代价
  • 三、标准库中的阻塞队列
    • (一)观察模型的运行效果
    • (二)观察阻塞效果
      • 1) 队列为空的阻塞效果
      • 2) 队列为满的阻塞效果
  • 四、模拟实现阻塞队列
    • 1) 实现要点
    • 2) 基于数组实现普通队列
    • 3) 添加所需字段
    • 4) 循环队列逻辑
    • 5) 实现 put()
    • 6) 实现take()
    • 7) 对 put 和 take 实现阻塞效果

一、阻塞队列

在数据结构学习集合类是我们接触了队列、优先级队列,都是一些很重要的数据结构,尤其是现在搞后端开发,经常会使用分布式系统,微服务框架等等

  • 阻塞队列,是一种更加复杂的队列
  • 也遵守"先进先出"的原则
  • 阻塞队列是一种线程安全的数据结构
  • 阻塞特性:
    1. 队列为空时,尝试出队列,出队列操作就会阻塞,阻塞到其他线程添加元素为止
    1. 队列为满时,尝试入队列,入队列操作就会阻塞,阻塞到其他线程取走元素为止

阻塞队列的一个典型应用场景就是"生产者消费者模型",这是一种非常典型的开发模型

在这里插入图片描述

二、生产者消费者模型

(一)概念

  • 生产者消费者模型就是通过一个容器来解决生产者和消费者的强耦合问题
  • 生产者和消费者彼此之间不直接通讯,而是通过阻塞队列来进行通讯,所以生产者生产完数据之后不用等待消费者处理,直接扔给阻塞队列,消费者不找生产者要数据,而是直接从阻塞队列里取

对于上述描述,举个例子,方便理解~~
请出我们的老三位,朝新,朝望,小舟,三个人包饺子

  • 两个包饺子的方案
    1. 三个人,每个人都分别擀一个饺子皮,包一个饺子
  • 这个方案三个线程就会竞争同一个资源 =>擀面杖,造成阻塞等待
    1. 朝新专门负责擀饺子皮,另外两个人负责包饺子

在这里插入图片描述

  • 当朝新擀饺子皮,擀得飞起~~,会出现阻塞太多了,消费者来不及消费,造成桌子上没地方放饺子皮了,就阻塞了
  • 诶? 上面第一个方案就是因为阻塞等待才pass的,这里产生阻塞怎么就行呢?

这里我们提到的阻塞是一个"极端情况",生产者消费者之间速度不协调才会出现,一般情况下,都是不会出现阻塞的

(二)生产者消费者的两个重要优势(阻塞队列的运用)

    1. 解耦合 : 阻塞队列也能是生产者和消费者之间解耦,降低耦合是为了让后续修改的时候成本低~
    1. 削峰填谷 : 阻塞队列相当于一个缓冲区,平衡了生产者和消费者的处理能力

比如在"秒杀"场景下,服务器同一时刻可能会收到大量的支付请求,如果直接处理这些支付请求,服务器可能扛不住(每个支付请求处理都需要比较复杂的流程),这个时候就可以把这些请求都放到一个阻塞队列中,然后再由消费者线程慢慢的按照自己的节奏来处理每个支付请求,不至于下游服务器直接崩溃
这样做可以有效进行"削峰",防止服务器被突然到来的一波请求直接冲垮

1) 解耦合(不一定是两个线程之间,也可以是两个服务器之间)

在这里插入图片描述

当我们在两个服务器之间引入 阻塞队列后

在这里插入图片描述

  • 本来是 A 和 B耦合,现在是 A和队列耦合,B和队列耦合
  • 同样是耦合,为什么单单与队列耦合,就是我们所希望的呢?
  • 因为队列的作用基本上是入队列,出队列,功能单一,固定,一般不会涉及到修改,大大降低了耦合,后续修改服务器代码,成本低

2) 削峰填谷

在这里插入图片描述
这张图可以理解为服务器收到的请求量的曲线图

在这里插入图片描述

  • 一般来说 A这种上游的服务器,尤其是 入口的服务器,干的活更简单,单个请求消耗的资源数少

  • 像 B这种下游的服务器,同行承担更重的任务量,复杂的计算/存储 工作,单个请求消耗的资源数更多

  • 日常工作中,确实会给 B这样角色的服务器分配更好的机器,即使如此,也很难保证 B 承担的访问量能够比 A更高

  • 服务器会什么会挂呢? 比如每次选课时,选课系统就会卡的进不去

  • 服务器每个请求的时候,都是需要消耗一定的硬件资源

  • 包括不限于 CPU,内存,硬盘,网络带宽的资源

  • 同时有N个请求呢? 消耗的量*N

  • 一旦消耗的总量,超出了机器硬件资源的上限,此时,对应的进程就可能崩溃或者操作系统产生卡顿 =>挂了

  • 提供的量 < 消耗的量 ,就会挂~

当我们在两个服务器之间引入 阻塞队列后

在这里插入图片描述

  • 一般来说,请求量激增是突发,时间也会短
  • 趁着峰值过去了,B 仍然继续消费数据,利用波谷的时间,阿里赶紧消费之前积压的数据

(三)生产者消费者模型付出的代价

    1. 引入队列后,整体的结构会更复杂,此时,就需要更多的机器,进行部署,生产环境的结构会更复杂,管理起来更加麻烦
    1. 运用阻塞队列,导致性能降低,效率会有影响

在这里插入图片描述

三、标准库中的阻塞队列

在Java标准库中内置了阻塞队列,如果我们需要在一些程序中使用阻塞队列,直接使用标准库中的即可

  • BlockingQueue 是一个接口,真正实现的类是 LinkedBlockingQueue
  • put方法用于阻塞式的入队列,take用于阻塞式出队列
  • BlockingQueue 也有offer ,poll,peek等方法,但是这些方法不带有阻塞特性

在这里插入图片描述

public class Demo29 {
    public static void main(String[] args) throws InterruptedException{
        BlockingQueue<String> queue = new LinkedBlockingQueue<>(100);
        for (int i = 0; i < 100;i++){
            queue.put("a");
        }
        System.out.println("队列已满");
        queue.put("a");
        System.out.println("再次尝试 put 元素");
    }
}

使用jconsole观察线程状态

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

  • 其中我们添加了阻塞队列的参数capacity =>最多能容纳多少元素
  • 如果不设置capacity,默认是一个非常大的值
  • 实际开发中,一般建议大家能够设置上要求的最大值
  • 否则,队列可能变得非常大,导致把内存耗尽,产生内存超出范围这样的异常

在这里插入图片描述

  • 填满也没事,队列最多21亿个数据,每个元素是一个int(4个字节)
  • 极端情况 80亿个字节 => 8G,打满了消耗 8GB内存
  • 我们电脑都是 16G 32G 倒是能消耗得起
  • 不过会影响效率,最好还是设置capacity
  • 一个JVM进程也不一定能够利用机器所有的内存,是可以在运行JVM的时候通过一定的参数指定JVM最多消耗多少内存
  • 如果实际消耗的内存,超过了JVM运行时候的限制上限,确实会挂~
  • Thousand 千 => K
  • Million 百万 => M
  • Billion 十亿 => G

(一)观察模型的运行效果

public class Demo30 {
    // 生产者一个线程  消费者一个线程
    public static void main(String[] args) {
        BlockingQueue<Integer> queue = new LinkedBlockingQueue<>();
        Thread producer = new Thread(() -> {
            int n = 0;
            while (true) {
                try {

                    queue.put(n);
                    //Thread.sleep(1000);
                } catch (InterruptedException e) {
                    throw new RuntimeException(e);
                }
                System.out.println("生产元素 " + n);
                n++;
            }
        }, "producer");
        Thread consumer = new Thread(() -> {
            while(true){
                try {
                    Integer n = queue.take();
                    System.out.println("消费元素 " + n);
                    //Thread.sleep(1000);
                } catch (InterruptedException e) {
                    throw new RuntimeException(e);
                }
            }
        }, "consumer");
        producer.start();
        consumer.start();
    }
}

在这里插入图片描述

  • 通过打印的日志信息,可以观察到,两个线程的执行速度旗鼓相当,并没有产生阻塞
  • 这就是开发中的典型情况,虽然模型是会出现阻塞的,但是只要我们协调好生产和消费的速度,两个线程执行速度相差不大,程序就会高效的运行

(二)观察阻塞效果

1) 队列为空的阻塞效果

  • 上述的 producer 和 consumer 两个线程的速度旗鼓相当,很难观察到阻塞
  • 我们添加sleep,使得两线程速度相差较大,来观察阻塞队列产生的阻塞效果

在这里插入图片描述
通过降低producer调用 put() 的速度,阻塞队列中的元素被消耗的速度远远大于生产的速度,进而使阻塞队列对 consumer的take() 产生阻塞效果

2) 队列为满的阻塞效果

  • 降低 consumer的消费速度,观察阻塞队列对producer的阻塞效果

在这里插入图片描述

  • 虽然 producer 只执行了1秒,但已经把阻塞队列填满了
  • 因为队列已满,producer的put()方法产生阻塞效果
  • 所以只能consumer消费一个元素,生产者才能生产一个元素

四、模拟实现阻塞队列

我们自己模拟实现一个简单的阻塞队列,并且基于这个阻塞队列实现 生产者消费者模型

1) 实现要点

  • 通过"循环队列"的方式来实现
  • 使用synchronized进行加锁控制
  • put 插入元素时,判定如果队列满了,就进行wait
  • take 取出元素时,判定如果队列为空,就进行wait
  • wait的使用需要注意,建议配合while使用(后面详细介绍)

2) 基于数组实现普通队列

class MyBlockingQueue{
    private  String[] data = null;
    
    public MyBlockingQueue(int capacity){
        data = new String[capacity];
    }
}

public class Demo31 {
    public static void main(String[] args) {
        
    }
}

3) 添加所需字段

class MyBlockingQueue{
    private  String[] data = null;

    private int head = 0;//队头

    private int tail = 0;//队尾

    private int size = 0;//元素个数

    public MyBlockingQueue(int capacity){
        data = new String[capacity];
    }
}

4) 循环队列逻辑

  • put()元素 放入tail处,take()元素 head处取出
  • put()元素 => tail++
  • take()元素 =>head++
  • 若 head 和 tail > data.length =>两个指针置为0,继续循环
  • 若 head 和 tail 指向同一个元素,要么队空,要么队满
  • 要么 所有线程阻塞在 put,要么所有线程阻塞在take

5) 实现 put()

  • 队满要进入阻塞(如何进入阻塞)
  • put()一次,tail++ size ++
  • tail 走出队列 要置为0
 public void put(String elem){
        if(size == data.length){
            //队列满了,进入阻塞
            return;
        }
        data[tail] = elem;
        tail++;
        if(tail >= data.length){
            tail = 0;
        }
        size++;
    }
}

6) 实现take()

  • 队空要进入阻塞(如何进入阻塞)
  • take()一次 head++ size–
  • head走出队列 要置为0
public  String take(){
        if(size == 0){
            //队列为空,进入阻塞
            return null;
        }
        String ret = data[head];
        head++;
        if(head == data.length){
            head = 0;
        }
        size--;
        return ret;
    }
}

7) 对 put 和 take 实现阻塞效果

在这里插入图片描述

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

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

相关文章

《100天精通Python——基础篇 2025 第5天:巩固核心知识,选择题实战演练基础语法》

目录 一、踏上Python之旅二、Python输入与输出三、变量与基本数据类型四、运算符五、流程控制 一、踏上Python之旅 1.想要输出 I Love Python,应该使用()函数。 A.printf() B.print() C.println() D.Print() 在Python中想要在屏幕中输出内容&#xff0c;应该使用print()函数…

机器人夹爪的选型与ROS通讯——机器人抓取系统基础系列(六)

文章目录 前言一、夹爪的选型1.1 任务需求分析1.2 软体夹爪的选型 二、夹爪的ROS通讯2.1 夹爪的通信方式介绍2.2 串口助手测试2.3 ROS通讯节点实现 总结Reference: 前言 本文将介绍夹爪的选型方法和通讯方式。以鞋子这类操作对象为例&#xff0c;将详细阐述了对应的夹爪选型过…

第二十八章 RTC——实时时钟

第二十八章 RTC——实时时钟​​​​​​​ 目录 第二十八章 RTC——实时时钟 1 RTC实时时钟简介 2 RTC外设框图剖析 3 UNIX时间戳 4 与RTC控制相关的库函数 4.1 等待时钟同步和操作完成 4.2 使能备份域涉及RTC配置 4.3 设置RTC时钟分频 4.4 设置、获取RTC计数器及闹钟 5 实时时…

使用 DuckLake 和 DuckDB 构建 S3 数据湖实战指南

本文介绍了由 DuckDB 和 DuckLake 组成的轻量级数据湖方案&#xff0c;旨在解决传统数据湖&#xff08;如HadoopHive&#xff09;元数据管理复杂、查询性能低及厂商锁定等问题。该方案为中小规模数据湖场景提供了简单、高性能且无厂商锁定的替代选择。 1. 什么是 DuckLake 和 D…

大语言模型提示词(LLM Prompt)工程系统性学习指南:从理论基础到实战应用的完整体系

文章目录 前言&#xff1a;为什么提示词工程成为AI时代的核心技能一、提示词的本质探源&#xff1a;认知科学与逻辑学的理论基础1.1 认知科学视角下的提示词本质信息处理理论的深层机制图式理论的实际应用认知负荷理论的优化策略 1.2 逻辑学框架下的提示词架构形式逻辑的三段论…

如何基于Mihomo Party http端口配置git与bash命令行代理

如何基于Mihomo Party http端口配置git与bash命令行代理 1. 确定Mihomo Party http端口配置 点击内核设置后即可查看 默认7892端口&#xff0c;开启允许局域网连接 2. 配置git代理 配置本机代理可以使用 127.0.0.1 配置局域网内其它机代理需要使用本机的非回环地址 IP&am…

埃文科技智能数据引擎产品入选《中国网络安全细分领域产品名录》

嘶吼安全产业研究院发布《中国网络安全细分领域产品名录》&#xff0c;埃文科技智能数据引擎产品成功入选数据分级分类产品名录。 在数字化转型加速的今天&#xff0c;网络安全已成为企业生存与发展的核心基石&#xff0c;为了解这一蓬勃发展的产业格局&#xff0c;嘶吼安全产业…

NLP学习路线图(二十六):自注意力机制

一、为何需要你&#xff1f;序列建模的困境 在你出现之前&#xff0c;循环神经网络&#xff08;RNN&#xff09;及其变种LSTM、GRU是处理序列数据&#xff08;如文本、语音、时间序列&#xff09;的主流工具。它们按顺序逐个处理输入元素&#xff0c;将历史信息压缩在一个隐藏…

Unity3D仿星露谷物语开发60之定制角色其他部位

1、目标 上一篇中定制了角色的衬衫、手臂。 本篇中将定制角色其他部位的图形&#xff0c;包括&#xff1a;裤子、发型、皮肤、帽子等。 2、定制裤子 &#xff08;1&#xff09;修改ApplyCharacterCustomisation.cs脚本 我们需要设置一个输入框选择裤子的颜色。 // Select …

Google机器学习实践指南(机器学习模型泛化能力)

&#x1f525; Google机器学习(14)-机器学习模型泛化能力解析 Google机器学习(14)-机器学习模型泛化原理与优化&#xff08;约10分钟&#xff09; 一、泛化问题引入 ▲ 模型表现对比&#xff1a; 假设森林中树木健康状况预测模型&#xff1a; 图1&#xff1a;初始模型表现 …

MySQL性能调优:Mysql8高频面试题汇总

1&#xff0c;主键和唯一键有什么区别&#xff1f; 主键不能重复&#xff0c;不能为空&#xff0c;唯一键不能重复&#xff0c;可以为空。 建立主键的目的是让外键来引用。 一个表最多只有一个主键&#xff0c;但可以有很多唯一键 2&#xff0c;MySQL常用的存储引擎有哪些&…

vue+elementUI+springboot实现文件合并前端展示文件类型

项目场景&#xff1a; element的table上传文件并渲染出文件名称点击所属行可以查看文件,并且可以导出合并文件,此文章是记录合并文档前端展示的帖子 解决方案&#xff1a; 后端定义三个工具类 分别是pdf,doc和word的excle的目前我没整 word的工具类 package com.sc.modules…

高效绘制业务流程图!专业模板免费下载

在复杂的业务流程管理中&#xff0c;可视化工具已成为提升效能的核心基础设施。为助力开发者、项目经理及业务架构师高效落地流程标准化&#xff0c;本文将为你精选5套开箱即用的专业流程图模板。这些模板覆盖跨部门协作、电商订单、客户服务等高频场景&#xff0c;具备以下核心…

Spring Boot + Prometheus 实现应用监控(基于 Actuator 和 Micrometer)

文章目录 Spring Boot Prometheus 实现应用监控&#xff08;基于 Actuator 和 Micrometer&#xff09;环境准备示例结构启动和验证验证 Spring Boot 应用Prometheus 抓取配置&#xff08;静态方式&#xff09;Grafana 面板配置总结 Spring Boot Prometheus 实现应用监控&…

PowerBI企业运营分析—列互换式中国式报表分析

PowerBI企业运营分析—列互换式中国式报表分析 欢迎来到Powerbi小课堂&#xff0c;在竞争激烈的市场环境中&#xff0c;企业运营分析平台成为提升竞争力的核心工具。 该平台通过高效整合多源数据&#xff0c;并实时监控关键指标&#xff0c;能够迅速揭示业务表现的全貌&#…

BugKu Web渗透之需要管理员

启动场景&#xff0c;打开网页&#xff0c;显示如下&#xff1a; 一般没有上面头绪的时候&#xff0c;就是两步&#xff1a;右键查看源代码 和 扫描网站目录。 步骤一&#xff1a; 右键查看源代码 和 扫描网站目录。 右键查看源代码没有发现异常。 于是扫描网站目录&…

TDengine 开发指南—— UDF函数

UDF 简介 在某些应用场景中&#xff0c;应用逻辑需要的查询功能无法直接使用内置函数来实现&#xff0c;TDengine 允许编写用户自定义函数&#xff08;UDF&#xff09;&#xff0c;以便解决特殊应用场景中的使用需求。UDF 在集群中注册成功后&#xff0c;可以像系统内置函数一…

使用vsftpd搭建FTP服务器(TLS/SSL显式加密)

安装vsftpd服务 使用vsftpd RPM安装包安装即可&#xff0c;如果可以访问YUM镜像源&#xff0c;通过dnf或者yum工具更加方便。 yum -y install vsftpd 启动vsftpd、查看服务状态 systemctl enable vsftpd systemctl start vsftpd systemctl status vsftpd 备份配置文件并进…

1.1Nodejs和浏览器中的二进制处理

Buffer 在 Node.js 中&#xff0c;Buffer 类用于处理二进制数据。由于 JavaScript 在浏览器环境中主要用于处理字符串和数字等类型的数据&#xff0c;对二进制数据的处理能力较弱&#xff0c;因此 Node.js 引入了 Buffer 类来弥补这一不足&#xff0c;特别是在处理文件系统操作…

入门AJAX——XMLHttpRequest(Post)

一、前言 在上篇文章中&#xff0c;我们已经介绍了 HMLHttpRequest 的GET 请求的基本用法&#xff0c;并基于我提供的接口练习了两个简单的例子。如果你还没有看过第一篇文章&#xff0c;强烈建议你在学习完上篇文章后再学习本篇文章&#xff1a; &#x1f517;入门AJAX——XM…