当前位置: 首页 > news >正文

JavaEE多线程案例之阻塞队列

上文我们了解了多线程案例中的单例模式,此文我们来探讨多线程案例之阻塞队列吧

1. 阻塞队列是什么?

阻塞队列是⼀种特殊的队列.也遵守"先进先出"的原则.
阻塞队列是⼀种线程安全的数据结构,并且具有以下特性:

  • 当队列满的时候,继续⼊队列就会阻塞,直到有其他线程从队列中取⾛元素.
  • 当队列空的时候,继续出队列也会阻塞,直到有其他线程往队列中插⼊元素.

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

那么什么是生产者消费者模型呢??

1.1 生产者消费者模型

1. 生产者消费者模式就是通过⼀个容器来解决生产者和消费者的强耦合问题

生产者和消费者彼此之间不直接通讯,而是通过阻塞队列来进行通讯,所以生产者生产完数据之后不用等待消费者处理,直接扔给阻塞队列,消费者不找⽣产者要数据,而是直接从阻塞队列⾥取.

应用场景:
一对夫妻开包子店,早上夫妻二人包包子,女的负责擀包子皮(擀好的包子皮放在桌板上(类似于阻塞队列)),男的负责包包子,其中擀包子皮的就是生产者,包包子的就是消费者。擀包子皮的人不在意谁消耗他生产出来的包子皮,包包子的人也不在意是谁生产的包子皮,能用就可以

2. 阻塞队列就相当于⼀个缓冲区,平衡了⽣产者和消费者的处理能⼒. (削峰填⾕)

应用场景:
在每年的双十一、双十二购物节,服务器同⼀时刻可能会收到大量的支付请求,如果同时处理这些支付请求服务器可能扛不住(每个支付请求的处理都需要比较复杂的流程).
这个时候就可以把这些请求都放到⼀个阻塞队列中(类似于上面提到的放包子皮的桌板,消费者不需要按照生产者的请求速度来完成,可以按照自己的速度来完成,就保证了不会服务器崩掉),
然后再由消费者线程慢慢的来处理每个⽀付请求.这样做可以有效进⾏ “削峰”, 防⽌服务器被突然到来的⼀波请求直接冲垮.

3. 阻塞队列可以实现异步操作

1.2 标准库中的阻塞队列

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

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

在这里插入图片描述

1.3 消息队列的自我实现

  1. 首先我们采用的是循环队列的方式实现消息队列

定义一个数组用于存放消息
定义一个头尾指针标记消息队列的队头和队尾
定义一个size用于标记有效数据个数

//定义一个消息队列数组
private Integer [] information=null;
//定义一个头指针
private int head=0;
//尾指针
private int tail=0;
//定义一个size用于标记有效数据个数
private int size=0;
  1. 此时我们数组不给定大小,我们采用构造方法在初始化时,给定数组大小,此时和JDK实现的雷同
//构造方法用于初始化消息队列的容量大小
public MyBlockQueue(Integer capacity) {if(capacity<=0){throw new RuntimeException("队列容量必须大于0.");}information=new Integer[capacity];
}

在这里插入图片描述

  1. 在队尾插入元素

为了实现消息队列的效果,就要与普通队列不一样 在插入元素时,
若消息队列满了,就等待,等到消息队列有空位时,就会被唤醒
当插入元素时,就唤醒出元素的的操作
其中,wait()方法、notifyAll()需要搭配synchronize关键字使用

public void put(int value) throws InterruptedException {//没有位置就要等待if(size>=information.length){synchronized (this){this.wait();}}//有位置直接插入元素,无需等待if(size<information.length){information[tail]=value;tail++;size++;if(tail>=information.length){tail=0;}synchronized (this){this.notifyAll();}}}
  1. 在队头出元素

为了实现消息队列的效果,当消息队列中没有元素时,就需要等待,直到有元素进来,再被唤醒
当出元素时,有位置空余出来,就可以唤醒要插入的操作

//取出元素public int take() throws InterruptedException {//队列为空,需要等待if(size==0){synchronized (this){this.wait();}}//有元素直接取出即可Integer value=information[head];head++;size--;if(head==information.length){head=0;}synchronized (this){this.notifyAll();}return value;}
  1. synchronize关键字实现了原子性、在效果上实现了内存可见性,但是没有保证有序性,所以我们为涉及修改的变量加上volatile关键字
    //定义一个消息队列数组private volatile Integer [] information=null;//定义一个头指针private volatile int head=0;//尾指针private volatile int tail=0;//定义一个size用于标记有效数据个数private volatile int  size=0;
  1. 由于在插入和删除的操作中涉及多个变量的修改,我们可以扩大synchronize的范围

此时,我们完整的代码如下:

public class MyBlockQueue {//定义一个消息队列数组private volatile Integer [] information=null;//定义一个头指针private volatile int head=0;//尾指针private volatile int tail=0;//定义一个size用于标记有效数据个数private volatile int  size=0;//构造方法用于初始化消息队列的容量大小public MyBlockQueue(Integer capacity) {if(capacity<=0){throw new RuntimeException("队列容量必须大于0.");}information=new Integer[capacity];}//插入元素public void put(int value) throws InterruptedException {synchronized (this){//没有位置就要等待if(size>=information.length){this.wait();}//有位置直接插入元素,无需等待if(size<information.length){information[tail]=value;tail++;size++;if(tail>=information.length){tail=0;}synchronized (this){this.notifyAll();}}}}//取出元素public int take() throws InterruptedException {synchronized (this){//队列为空,需要等待if(size==0){synchronized (this){this.wait();}}//有元素直接取出即可Integer value=information[head];head++;size--;if(head==information.length){head=0;}this.notifyAll();return value;}}
}    
  1. 此时引发一个新问题,假设此时有多个线程要进行put操作,但只有一个空余位置,会有什么问题吗

在这里插入图片描述
所以我们将if判断改成while判断,并且在JDK提供的put方法中也是用while
在这里插入图片描述

1.4 消息队列代码

public class MyBlockQueue {//定义一个消息队列数组private volatile Integer [] information=null;//定义一个头指针private volatile int head=0;//尾指针private volatile int tail=0;//定义一个size用于标记有效数据个数private volatile int  size=0;//构造方法用于初始化消息队列的容量大小public MyBlockQueue(Integer capacity) {if(capacity<=0){throw new RuntimeException("队列容量必须大于0.");}information=new Integer[capacity];}//插入元素public void put(int value) throws InterruptedException {synchronized (this){//此处最好使⽤ while.//否则notifyAll 的时候, 该线程从wait 中被唤醒,  但是紧接着并未抢占到锁.//当锁被抢占的时候, 可能⼜已经队列满了while (size>=information.length){this.wait();}information[tail]=value;tail++;size++;if (tail>information.length){tail=0;}this.notifyAll();}}//取出元素public int take() throws InterruptedException {synchronized (this){//队列为空,需要等待while (size==0){this.wait();}//有元素直接取出即可Integer value=information[head];head++;size--;if(head==information.length){head=0;}this.notifyAll();return value;}}
}

希望得到你的支持,谢谢!

相关文章:

JavaEE多线程案例之阻塞队列

上文我们了解了多线程案例中的单例模式&#xff0c;此文我们来探讨多线程案例之阻塞队列吧 1. 阻塞队列是什么&#xff1f; 阻塞队列是⼀种特殊的队列.也遵守"先进先出"的原则. 阻塞队列是⼀种线程安全的数据结构,并且具有以下特性: 当队列满的时候,继续⼊队列就会…...

梳理你的思路(从OOP到架构设计)_基本OOP知识04

目录 1、 主动型 vs.基於被动型 API 1&#xff09;卡榫函数实现API 2&#xff09;API的分类 3&#xff09;回顾历史 4&#xff09;API >控制力 2、 结语&复习&#xff1a; 接口与类 1&#xff09;接口的表示 2&#xff09;Java的接口表示 1、 主动型 vs.基於被动…...

nginx反向代理(负载均衡)

nginx的代理 代理 四层代理 七层代理 正向代理和缓存的配置方式 &#x1f42d;&#x1f42e;&#x1f42f;&#x1f430;&#x1f409;&#x1f40d;&#x1f434;&#x1f411;&#x1f412;&#x1f414;&#x1f436;&#x1f437; 反向代理》负载均衡 负载均衡&#xff…...

Android系统应用主要模块

设置 Android Settings开发总结 Launcher Android Launcher开发学习总结 System UI Android SystemUI 学习总结...

【万字详解】三维重建(二)——NeRF、NeuS、MeshUDF、NeuralUDF、3DGS、GShell

文章目录 一、NeRF:Representing Scenes as Neural Radiance Fields for View Synthesis(推荐读)1.1 式1 神经网络的输入和输出1.2 式2 体素渲染算法1.3 式3 损失函数1.4 位置编码1.5 基本原理二、经典的重建流程2.1 传统的三维重建pipeline2.2 神经网络回归2.3 可微渲染最优…...

【RK3588 Linux 5.x 内核编程】-内核线程与Seqlock

内核线程与Seqlock 文章目录 内核线程与Seqlock1、Seqlock介绍2、Seqlock相关API2.1 初始化2.2 写操作2.3 读操作3、驱动实现4、驱动验证在前面的文章中,我们介绍了 Mutex、Spinlock、Read/Write Spinlock 的使用及其实现。 它们都用于保护共享资源不被两个或多个进程同时修改…...

访问者模式的理解和实践

在软件开发过程中&#xff0c;设计模式为我们提供了解决常见问题的最佳实践。访问者模式&#xff08;Visitor Pattern&#xff09;是行为设计模式之一&#xff0c;它将数据操作与数据结构分离&#xff0c;使得在不修改数据结构的前提下&#xff0c;能够定义作用于这些元素的新的…...

在Scala中对Map函数的使用

package pp28{object cscc {def main(args: Array[String]): Unit {val m1 Map("马云 — 阿里巴巴" -> 1964,"马化腾 — 腾讯" -> 1971,"李彦宏 - 百度" -> 1968,"雷军 - 小米" -> 1969,"丁磊 - 网易" -> …...

PyTorch基本使用-张量的索引操作

在操作张量时&#xff0c;经常要去获取某些元素进行处理或者修改操作&#xff0c;在这里需要了解torch中的索引操作。 准备数据&#xff1a; data torch.randint(0,10,[4,5]) print(data--->,data)输出结果&#xff1a; data---> tensor([[3, 9, 4, 0, 5],[7, 5, 9, …...

OpenCV实验:图片加水印

第二篇&#xff1a;图片添加水印&#xff08;加 logo&#xff09; 1. 实验原理 水印原理&#xff1a; 图片添加水印是图像叠加的一种应用&#xff0c;分为透明水印和不透明水印。水印的实现通常依赖于像素值操作&#xff0c;将水印图片融合到目标图片中&#xff0c;常用的方法…...

sql server log文件

确定 SQL Server 实例中具有大量 VDF 的数据库 SELECT [name], COUNT(l.database_id) AS vlf_count FROM sys.databases AS s CROSS APPLY sys.dm_db_log_info(s.database_id) AS l GROUP BY [name] HAVING COUNT(l.database_id) > 100; 在收缩日志文件之前确定事务日志中…...

Elasticsearch 集群部署

Elasticsearch 是一个分布式的搜索和分析引擎&#xff0c;广泛应用于日志分析、全文搜索、实时数据分析等场景。它以其高性能、高可用性和易用性而著称。本文档将引导您完成一个基本的 Elasticsearch 集群配置&#xff0c;包括节点间的通信、客户端访问、安全设置等关键步骤。我…...

微信小程序5-图片实现点击动作和动态加载同类数据

搜索 微信小程序 “动物觅踪” 观看效果 感谢阅读&#xff0c;初学小白&#xff0c;有错指正。 一、功能描述 a. 原本想通过按钮加载背景图片&#xff0c;来实现一个可以点击的搜索button&#xff0c;但是遇到两个难点&#xff0c;一是按钮大小调整不方便&#xff08;网上搜索…...

策略梯度定理公式的详细推导

策略梯度定理公式的详细推导 以下是策略梯度定理公式从基础概率公式到最终形式的完整推导&#xff0c;帮助更清晰地理解推导过程中的每一个步骤。 1. 策略梯度的目标 我们希望最大化期望累积奖励 ( J ( θ ) J(\theta) J(θ) )&#xff0c;其定义为&#xff1a; J ( θ ) E…...

力扣-图论-10【算法学习day.60】

前言 ###我做这类文章一个重要的目的还是给正在学习的大家提供方向和记录学习过程&#xff08;例如想要掌握基础用法&#xff0c;该刷哪些题&#xff1f;&#xff09;我的解析也不会做的非常详细&#xff0c;只会提供思路和一些关键点&#xff0c;力扣上的大佬们的题解质量是非…...

《Python WEB安全 库全解析》

《Python WEB安全 库全解析》 一、Python WEB安全 库概述二、常见的 Python WEB安全 库介绍1. Jiasule2. Awesome Python Security3. Flask-Security4. Flask-SeaSurf 三、Python WEB 安全库的优缺点1. 优点2. 缺点 四、Python WEB 安全库的使用场景1. 开发 Web 应用2. 处理敏感…...

Linux yum-config-manager命令异常

错误信息 使用 yum-config-manager命令时错误信息如下 sudo yum-config-manager \ > --add-repo \ > https://download.docker.com/linux/centos/docker-ce.repo sudo: yum-config-manager: command not found 解决办法 第一步&#xff1a; sudo yum -y install yum-u…...

ios 开发配置蓝牙

如果使用了蓝牙功能, 又没有配置, 会出现以下错误: This app has crashed because it attempted to access privacy-sensitive data without a usage description. The apps Info.plist must contain an NSBluetoothAlwaysUsageDescription key with a string value explaini…...

geoserver(1) 发布sql 图层 支持自定义参数

前提使用postgis 数据库支持关联 join 支持 in,not in,like,及其他sql原生函数 新增sql图层 编写自定义sql 编辑sql语句必须输出带有geom数据 正则表达式去除 设置id以及坐标参考系 预览sql图层效果 拼接sql参数 http://xxx.com/geoserver/weather/wms?SERVICEWMS&VERSI…...

Linux:network:添加ip的时候自动添加一个本地路由

文章目录 问题问题 最近在看一个路由的问题,顺便看内核代码,发现在添加IP的时候,内核会自动添加一个local route。 net/ipv4/devinet.c inet_rtm_newaddr->__inet_insert_ifa /* Send message first, then call notifier.Notifier will trigger FIB update, so thatlis…...

go 集成nacos注册中心、配置中心

使用限制 Go>v1.15 Nacos>2.x 安装 使用go get安装SDK&#xff1a; go get -u github.com/nacos-group/nacos-sdk-go/v2 快速使用 初始化客户端配置ClientConfig constant.ClientConfig{TimeoutMs uint64 // 请求Nacos服务端的超时时间&#xff0c;默…...

ssd202d-badblock-坏块检测

这边文章讲述的是坏快检测功能 思路&#xff1a; 1.第一次烧录固件会实现跳坏块&#xff0c;但是后续使用会导致坏块的产生&#xff1b; 于是我在uboot环境变量添加了两个变量来控制坏快 lb_badnum //坏块个数 lb_badoff //坏块所在位置 2.第一次开机会根据lb_badnum是否…...

MySQL-练习-数据介绍

文章目录 一. 数据介绍1. 数据结构2. 创建数据库,数据表3. 员工表(employees)练习1 4. 顾客表(customers)练习2 5. 商品(products)和商品类别(categories)表练习3 6. 供应商表&#xff08;suppliers&#xff09;练习4 7. 订单和订单明细表练习5 二. 数据汇总三. 使用CASE WHEN …...

React框架:解锁现代化Web开发的新维度

在当今前端开发领域&#xff0c;React 无疑是一颗璀璨的明星。React 是由 Facebook 开发的用于构建用户界面的 JavaScript 库&#xff0c;它在前端开发中占据着重要的地位&#xff0c;为开发者提供了一种高效、灵活且可维护的方式来构建复杂的用户界面。 一、React 的背景与开…...

电阻功率,限流,等效电阻

1 电阻额定功率 2 电阻限流作用 3 电阻并联等效电阻...

Qt | 开发工具(top1)

Qt Creator 跨平台、完整的集成开发环境(IDE)&#xff0c;供应用程序开发者创建用于多个桌面、嵌入式和移动设备平台的应用程序。 Qt Linguist 一套将Qt C和Qt Quick应用程序翻译成本地语言的工具。 qmake Qt自动化构建工具&#xff0c;简化了不同平台的构建过程。…...

Node.js express

1. express 介绍 express 是一个基于 Node.js 平台的极简、灵活的 WEB 应用开发框架&#xff0c;官方网址&#xff1a;https://www.expressjs.com.cn/简单来说&#xff0c;express 是一个封装好的工具包&#xff0c;封装了很多功能&#xff0c;便于我们开发 WEB 应用&#xff…...

ios h5中在fixed元素中的input被focus时,键盘遮挡input (van-popup、van-feild)

问题描述&#xff1a; 前提&#xff1a;我使用的是vant组件库&#xff0c;其中一个页面中有一个van-popup组件&#xff0c;van-popup组件中又嵌套了一个van-field组件预期结果&#xff1a;当点击van-feild输入框时&#xff0c;键盘弹起&#xff0c;输入框显示在键盘上方实际结…...

springboot整合lua脚本在Redis实现商品库存扣减

1、目的 使用lua脚本&#xff0c;可以保证多条命令的操作原子性&#xff1b;同时可以减少操作IO&#xff08;比如说判断redis对应数据是否小于0&#xff0c;小于0就重置为100&#xff0c;这个场景一般是取出来再判断&#xff0c;再存放进行&#xff0c;就至少存在2次IO,用lua脚…...

MySQL ON DUPLICATE KEY UPDATE影响行数

目录 分析为什么Updates返回7 总结 数据库更新日志如下 insertOrUpdateList|> Preparing: INSERT INTO clue_user_tag (vuid, tag_id, tag_type, content) VALUES (?, ?, ?, ?) , (?, ?, ?, ?) , (?, ?, ?, ?) , (?, ?, ?, ?) ON DUPLICATE KEY UPDATE …...

男女做暧昧小视频网站/公司网站开发费用

英文名称&#xff1a;TCO-PEG3-FITC 中文名称&#xff1a;反式环辛烯-三聚乙二醇-荧光素 分子式&#xff1a;C38H43N3O10S 分子量&#xff1a;733.83 纯度 gt;95% 外观&#xff1a;黄色固体 储存条件&#xff1a;-20C&#xff0c;避光避湿 用途&#xff1a;仅供科研使用…...

自己怎么做百度网站/深圳网站seo推广

迎接县均衡化国家验收学校解说词办学条件组尊敬的各位专家、各位领导&#xff1a;欢迎莅临我校检查指导工作。我们宁津县第二实验小学始建于1997年&#xff0c;是一所县属非寄宿完全小学。当时只有北面这一座楼&#xff0c;29名教师。2012年秋季扩建&#xff0c;建成南面这座教…...

制作网页和做网站是一个意思吗/网站查询服务器

本文欢迎转载&#xff0c;转载请注明出处和作者。由于各种监控系统的实时告警&#xff0c;都需要使用邮箱进行发送&#xff0c;而其中使用SMTP协议的邮箱&#xff0c;各种监控系统、ITSM系统等支持最广泛。为测试监控系统的告警功能&#xff0c;需要先搭建1个SMTP邮箱。实验环境…...

大型电商网站开发价格/热搜榜排名前十

SVM算法比较复杂&#xff0c;数学功底要求很高。 详见七月大神博客《 支持向量机通俗导论&#xff08;理解SVM的三层境界&#xff09;》转载于:https://www.cnblogs.com/ahu-lichang/p/7181773.html...

青岛公路建设集团有限公司网站/搜索引擎推广

JUnit是一个JAVA语言的单元测试框架。多数JAVA开发环境都己经集成了JUnit作为单元测试的工具。 一、 JUnit基本概念 1. TestCase:代表一个测试用例&#xff0c;每一个TestCase实例都对应一个测试&#xff0c;这个测试通过这个TestCase实例的名字标志同&#xff0c;以便在测试…...

旅游网站ppt应做的内容/手机怎么建立网站

一、表的关系分析&#xff1a;用户和订单&#xff1a;一个用户可以有多个订单&#xff0c;但每个订单只能属于一个用户&#xff0c;所以是一对多的关系。商品和分类&#xff1a;一个产品只能有一种分类&#xff0c;而一个分类可以有多种产品&#xff0c;所以是多对一的关系。订…...