【Java|多线程与高并发】阻塞队列以及生产者-消费者模型
阻塞队列(BlockingQueue)常用于多线程编程中,可以实现线程之间的同步和协作。它可以用来解决生产者-消费者问题,其中生产者线程将元素插入队列,消费者线程从队列中获取元素,它们之间通过阻塞队列进行协调。阻塞队列的使用可以简化多线程编程的复杂性,避免手动实现线程间的同步和协作逻辑,提高代码的可读性和可维护性。基于阻塞队列的生产者-消费者模型也要重点掌握.。阻塞队列作为生产者和消费者之间的缓冲
1. 前言
阻塞队列(BlockingQueue)常用于多线程编程中,可以实现线程之间的同步和协作。它可以用来解决生产者-消费者问题,其中生产者线程将元素插入队列,消费者线程从队列中获取元素,它们之间通过阻塞队列进行协调。
2. 阻塞队列
Java中的阻塞队列(BlockingQueue
)是一种特殊的队列,它在队列为空时会阻塞获取元素的操作,直到队列中有新的元素被添加进来;在队列已满时会阻塞插入元素的操作,直到队列中有空的位置。
需要注意的是 在Java中 BlockingQueue
是一个接口
Java中提供了多种阻塞队列的实现,包括:
- ArrayBlockingQueue:基于数组实现的有界阻塞队列,它按照先进先出的顺序对元素进行排序。
- LinkedBlockingQueue:基于链表实现的可选有界阻塞队列,它可以指定容量,如果不指定则默认为无界队列。
- PriorityBlockingQueue:基于优先级堆实现的无界阻塞队列,它可以按照元素的优先级进行排序。
- SynchronousQueue:一个不存储元素的阻塞队列,每个插入操作必须等待另一个线程的移除操作。
- LinkedBlockingDeque: 基于链表的双端阻塞队列。
- LinkedTransferQueue: 基于链表、无界的的阻塞队列。
作为一个队列,有三个基本操作,入队,出队和查看队首元素.
在使用阻塞队列时,有两点需要注意:
- offer和put可以实现入队列,offer并没有阻塞功能,put具有阻塞功能
- poll和take可以实现出队列,poll没有阻塞功能,take具有阻塞功能
示例:
public class Demo14 {
public static void main(String[] args) throws InterruptedException {
BlockingQueue<Integer> blockingQueue = new LinkedBlockingQueue<>(10);
blockingQueue.put(1);
blockingQueue.put(2);
int ret = blockingQueue.take();
System.out.println(ret);
ret = blockingQueue.take();
System.out.println(ret);
blockingQueue.take();
System.out.println(ret);
}
}
代码分析:
3. 实现阻塞队列
阻塞队列的实现并不复杂,主要是通过这个过程,强化对于阻塞队列的认识.
对于实现阻塞队列可以大致分为3步:
- 实现一个普通的队列
- 保证线程安全
- 加上阻塞功能
实现一个普通队列的方法可以使用链表,也能够使用数组.
接下来就基于数组来实现一个 循环队列
代码:
public class MyBlockingQueue {
private int[] elem;
private int usedSize = 0;// 有效个数
private int front = 0;
private int rear = 0;
public MyBlockingQueue(){
this.elem = new int[10];
}
public MyBlockingQueue(int k){
this.elem = new int[k];
}
/**
* 入队
* @param val
* @return
*/
public void put(int val){
// 判断队列是否满了
if (usedSize >= elem.length){
return;
}
// 进行入队列操作
elem[rear] = val;
rear++;
if (rear >= elem.length){
rear = 0;
}
usedSize++;
}
/**
* 出队
* @return
*/
public Integer take(){
if (usedSize == 0){
return null;
}
int ret = elem[front];
front++;
if (front >= elem.length){
front = 0;
}
usedSize--;
return ret;
}
}
循环队列在判断队列是否满时有两种方式:
-
用计数器记录有效数据的个数(上述代码使用的是这种方法)
-
浪费一个空间不用. 当
(rear+1) % 数组长度== front
时为队列满这种情况
以上就是一个简单的循环队列的实现,详细可以参考我的这篇文章【数据结构与算法】队列-模拟实现队列以及设计循环队列
解决了第一步,接下来就来实现第二步,解决线程安全问题
解决这个问题,就离不开synchronized
了,以防万一再给变量加上volatile
关键字
public class MyBlockingQueue {
private int[] elem;
private volatile int usedSize = 0;// 有效个数
private volatile int front = 0;
private volatile int rear = 0;
public MyBlockingQueue(){
this.elem = new int[10];
}
public MyBlockingQueue(int k){
this.elem = new int[k];
}
/**
* 入队
* @param val
* @return
*/
public void put(int val){
synchronized (this) {
// 判断队列是否满了
if (usedSize >= elem.length){
return;
}
// 进行入队列操作
elem[rear] = val;
rear++;
if (rear >= elem.length){
rear = 0;
}
usedSize++;
}
}
/**
* 出队
* @return
*/
public Integer take(){
synchronized (this) {
if (usedSize == 0){
return null;
}
int ret = elem[front];
front++;
if (front >= elem.length){
front = 0;
}
usedSize--;
return ret;
}
}
}
解决完线程安全问题,接下来就是给put和take添加阻塞功能
.
以下两种场景会有阻塞功能:
-
如果队列为空,出队列需要阻塞
-
如果队列为满,入队列需要阻塞
阻塞功能很好加,直接使用wait
方法即可
但除了阻塞,还要通知唤醒线程,. 例如线程为空,出队列在阻塞状态,而入队列之后,队列不为空,就要让出队列继续执行才行. 因此还要搭配notify
来实现
阻塞队列实现代码如下:
public class MyBlockingQueue {
private int[] elem;
private volatile int usedSize = 0;// 有效个数
private volatile int front = 0;
private volatile int rear = 0;
public MyBlockingQueue(){
this.elem = new int[10];
}
public MyBlockingQueue(int k){
this.elem = new int[k];
}
/**
* 入队
* @param val
* @return
*/
public void put(int val) throws InterruptedException {
synchronized (this) {
// 判断队列是否满了
while (usedSize >= elem.length){
this.wait();
}
// 进行入队列操作
elem[rear] = val;
rear++;
if (rear >= elem.length){
rear = 0;
}
usedSize++;
this.notify();
}
}
/**
* 出队
* @return
*/
public Integer take() throws InterruptedException {
synchronized (this) {
while (usedSize == 0){
this.wait();
}
int ret = elem[front];
front++;
if (front >= elem.length){
front = 0;
}
usedSize--;
this.notify();
return ret;
}
}
}
代码分析:
此时出队列和入队列是相互唤醒的状态, 两个wait的执行条件互斥,因此不会出现同时阻塞的状态.
同时将wait的执行条件改为while,是因为如果线程是被interrupt
唤醒的话,队列仍然为空,就不能去执行后续代码,因此要再进行条件判断,因此改为while更加稳妥
验证:
使用阻塞队列的好处:
- 使用阻塞队列,有利于代码"解耦合"
- 削峰填谷,利用生产者消费者模型在并发量高的时候将这些并发量分配到每一个服务器上.
4. 生产者-消费者模型
接下来来介绍生产者-消费者模型
生产者消费者模型是一种常见的并发编程模型,用于解决生产者和消费者之间的协作问题。在该模型中,生产者负责生产数据,消费者负责消费数据,它们通过共享的缓冲区来进行通信。
生产者消费者模型的主要特点:
- 生产者和消费者是两个独立的实体,它们可以运行在不同的线程中。
- 生产者负责生成数据,并将数据放入缓冲区中。
- 消费者负责从缓冲区中取出数据,并进行相应的处理。
- 缓冲区是生产者和消费者之间的共享数据结构,用于存储生产者生产的数据。
- 生产者和消费者之间的操作是互斥的,即同一时间只能有一个生产者或一个消费者对缓冲区进行操作。
为了实现生产者消费者模型,可以使用阻塞队列来作为缓冲区。为了实现生产者消费者模型,可以使用阻塞队列来作为缓冲区。
生产者消费者模型的基本流程:
- 创建一个共享的阻塞队列,作为生产者和消费者之间的缓冲区。
- 创建生产者线程,它生成数据并将其放入队列中。
- 创建消费者线程,它从队列中获取数据并进行处理。
- 启动生产者和消费者线程,它们会并发执行。
- 生产者线程生成数据并插入队列,当队列已满时会被阻塞。
- 消费者线程从队列中获取数据并进行处理,当队列为空时会被阻塞。
- 生产者和消费者线程可以通过阻塞队列进行同步和协作,生产者在队列已满时会等待,直到有空闲位置可以插入数据;消费者在队列为空时会等待,直到有数据可以获取。
代码示例:
public class Demo16 {
public static void main(String[] args) {
BlockingQueue<Integer> blockingQueue = new LinkedBlockingQueue<>();
// 生产者
Thread t1 = new Thread(() ->{
for (int i = 0; i < 1000; i++) {
try {
blockingQueue.put(i);
System.out.println("生产元素: "+ i);
Thread.sleep(200);
} catch (InterruptedException e) {
throw new RuntimeException(e);
}
}
});
t1.start();
// 消费者
Thread t2 = new Thread(() ->{
while(true){
try {
Integer ret = blockingQueue.take();
System.out.println("消费元素: "+ ret);
} catch (InterruptedException e) {
throw new RuntimeException(e);
}
}
});
t2.start();
}
}
运行截图:
上述代码仅仅只是一个示例,并没有什么实际意义.
生产者消费者模型依旧很重要,它可以有效地实现线程间的协作和资源共享,提高系统的并发性和吞吐量。
5. 总结
阻塞队列的使用可以简化多线程编程的复杂性,避免手动实现线程间的同步和协作逻辑,提高代码的可读性和可维护性。基于阻塞队列的生产者-消费者模型也要重点掌握.。阻塞队列作为生产者和消费者之间的缓冲区,提供线程安全的插入和获取操作,并在队列为空或队列已满时进行阻塞,从而实现线程间的同步。
感谢你的观看!希望这篇文章能帮到你!
专栏: 《从零开始的Java学习之旅》在不断更新中,欢迎订阅!
“愿与君共勉,携手共进!”
开放原子开发者工作坊旨在鼓励更多人参与开源活动,与志同道合的开发者们相互交流开发经验、分享开发心得、获取前沿技术趋势。工作坊有多种形式的开发者活动,如meetup、训练营等,主打技术交流,干货满满,真诚地邀请各位开发者共同参与!
更多推荐
所有评论(0)