news 2026/6/10 18:26:25

阻塞队列的使用和实现

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
阻塞队列的使用和实现

阻塞队列是一种特殊的队列,其遵循“先入先出”的原则。

阻塞队列也是一种线程安全的数据结构,具有以下特性:

  • 队列为满,入队列产生阻塞,直至其他线程从队列中取走元素
  • 队列为空,出队列产生阻塞,直至其他线程往队列中插入元素

“生产者消费者模型”是阻塞队列的一个典型应用场景,该模型也是一个典型的开发模型。

生产者消费者模型

生产者消费者模型就是通过一个中间容器来解决生产者和消费者之间的强耦合问题。

这个中间容器通过阻塞队列实现,从而使生产者和消费者之间不进行直接通讯。

阻塞队列的作用:

  1. 阻塞队列相当于一个缓冲区,平衡了生产者和消费者的处理能力(削峰填谷)
  2. 阻塞队列使生产者和消费者之间解耦

阻塞队列的缺点:

  1. 引入队列以后,代码整体结构变复杂
  2. 程序执行效率有所影响

阻塞队列的使用

Java的标准库中,提供了现成的阻塞队列。

  • BlockingQueue是一个接口,真正实现的类有:LinkedBlockingQueue(链表实现),ArrayBlockingQueue(数组实现),PriorityBlockingQueue(堆实现)等等。
  • 队列的出操作是poll,入队列操作是offer,但是阻塞队列使用的分别时take和put,这两个方法是带有阻塞功能的出入队列操作。
public class demo1 { public static void main(String[] args) { //创建阻塞队列 BlockingQueue<Integer> queue = new LinkedBlockingQueue<>(1000); //生产者线程 Thread producer = new Thread(() -> { int n = 0; while (true) { try { queue.put(n); System.out.println("生产元素 " + n); n++; } catch (InterruptedException e) { throw new RuntimeException(e); } } }, "procducer"); //消费者线程 Thread consumer = new Thread(() -> { while(true){ try { int n = queue.take(); System.out.println("消费元素 " + n); } catch (InterruptedException e) { throw new RuntimeException(e); } } }, "consumer"); producer.start(); consumer.start(); } }

阻塞队列的简单实现

  • 使用“循环队列”实现
  • 使用synchronized加锁保证线程安全
  • 注意这里的wait()搭配while使用,而不能搭配if使用,是由于notifyAll会将所有的wait唤醒,只有其中一个线程会put,而等到其他被唤醒的线程拿到锁之后,还需要确认一下容量是否已满,满的话还需要阻塞等待。
class BlockingQueue { private int[] item = new int[1000]; private volatile int head = 0; private volatile int tail = 0; private volatile int size = 0; //生产元素 public void put(int value) throws InterruptedException { synchronized (this) { while (size == item.length) { this.wait(); } item[tail] = value; tail++; if (tail == item.length) { tail = 0; } size++; notifyAll(); } } //消费元素 public int take() throws InterruptedException { synchronized (this) { while (size == 0) { this.wait(); } int ret = item[head]; head++; if(head == item.length){ head = 0; } size--; notifyAll(); return ret; } } //获取内部属性 public synchronized int getSize() { return size; } }
版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/6/10 0:48:14

ESP32-CAM UDP视频流实现:从零开始的实战案例

用ESP32-CAM打造低延迟UDP视频流&#xff1a;从电路到代码的完整实战指南你有没有试过用不到5美元的硬件&#xff0c;实现一个实时监控摄像头&#xff1f;不是买成品&#xff0c;而是自己动手从零搭建——从焊接引脚、烧录固件&#xff0c;到在电脑屏幕上看到第一帧由你控制的图…

作者头像 李华
网站建设 2026/6/10 13:58:38

ESP32-CAM图像捕获与上传:超详细版Arduino代码解析

用一块不到10美元的板子&#xff0c;实现远程拍照上传&#xff1a;ESP32-CAM实战全解析 你有没有想过&#xff0c;只花一杯咖啡的钱&#xff0c;就能做出一个能联网、会拍照、自动传图到服务器的小型监控设备&#xff1f;这并不是科幻电影里的桥段—— ESP32-CAM 就能做到。…

作者头像 李华
网站建设 2026/6/10 14:00:08

GeoJSON.io 免费在线地理编辑器:零基础快速上手终极教程

GeoJSON.io 免费在线地理编辑器&#xff1a;零基础快速上手终极教程 【免费下载链接】geojson.io A quick, simple tool for creating, viewing, and sharing spatial data 项目地址: https://gitcode.com/gh_mirrors/ge/geojson.io 还在为复杂的地理数据处理软件而头疼…

作者头像 李华
网站建设 2026/6/10 13:59:57

3分钟掌握抖音无水印下载:douyin_downloader终极教程

3分钟掌握抖音无水印下载&#xff1a;douyin_downloader终极教程 【免费下载链接】douyin_downloader 抖音短视频无水印下载 win编译版本下载&#xff1a;https://www.lanzous.com/i9za5od 项目地址: https://gitcode.com/gh_mirrors/dou/douyin_downloader 还在为抖音视…

作者头像 李华
网站建设 2026/6/10 13:57:01

NS模拟器一键安装神器:ns-emu-tools让小白也能轻松搞定

NS模拟器一键安装神器&#xff1a;ns-emu-tools让小白也能轻松搞定 【免费下载链接】ns-emu-tools 一个用于安装/更新 NS 模拟器的工具 项目地址: https://gitcode.com/gh_mirrors/ns/ns-emu-tools 还在为复杂的NS模拟器安装流程而头疼吗&#xff1f;ns-emu-tools作为一…

作者头像 李华