多线程学习(八)——线程通信之阻塞队列(BlockingQueue)控制线程通信(生产者、消费者问题)

本文主要是介绍多线程学习(八)——线程通信之阻塞队列(BlockingQueue)控制线程通信(生产者、消费者问题),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

        java5提供了一个BlockingQueue接口,虽然它也是Queue的子接口,但是它的主要用途不是作为容器,而是作为线程同步工具。BlockingQueue有一个特征:当生产者线程试图向BlockingQueue种放入元素的时候,如果该队列已满,则该线程被阻塞;当消费者线程试图从BlockingQueue种取元素的时候,如果该队列已空,则该线程被阻塞。

        程序的两个线程通过交替向BlockingQueue种放入元素、取出元素,即可以很好地控制线程的通信。BlockingQueue提供了两个方法来支持阻塞线程:

1、put(E e):尝试把E元素放入BlockingQueue中,如果该队列的元素已满,则阻塞该线程。

2、take():尝试从BlockingQueue的头部取出元素,如果该队列已空,则阻塞该线程。

BlockingQueue继承了Queue接口,也可以使用Queue接口中的方法,这些方法归纳起来如下:

BlockingQueue包含的方法之间的对应关系
 抛出异常不同返回值阻塞线程指定超时时长
队尾插入元素add(e)offer(e)put(e)offer(e,time,unit)
队头删除元素remove()poll()take()poll(time,unit)
获取、不删除元素element()peek()

BlockingQueue包含下面五个实现类:

1、ArrayBlockingQueue:基于数组实现的BlockingQueue

2、LinkedBlockingQueue:基于链表实现的BlockingQueue

3、PriorityBlockingQueue:并不是标准的阻塞队列,该队列调用remove()、poll()、take()等方法取出的元素,不是取出队列中存在时间最长的元素,而是队列中最小的元素

4、SynchronousBlockingQueue:同步队列。对该队列的存、取操作必须交替进行

5、DelayQueue:是一个特殊的BlockingQueue,底层基于 PriorityBlockingQueue 实现。不过DelayQueue要求集合元素都实现Delay接口(该接口中只有一个long getDelay()方法),DelayQueue根据集合元素的getDelay()方法的返回值进行排序。

利用BlockingQueue的特性,我们实现一段生产者,消费者的功能,代码如下面:

生产者类:

import java.util.concurrent.BlockingQueue;public class Producer extends Thread{private BlockingQueue<String> bq;public Producer(BlockingQueue<String> bq) {this.bq = bq;}public void run() {String[] strArr = new String[]{"java","hbase","spring"};for(int i=0;i<strArr.length;i++) {System.out.println(getName()+"生产者准备生产集合元素!");try {Thread.sleep(200);//尝试放入元素,如果队列已满,则线程被阻塞bq.put(strArr[i%3]);}catch(InterruptedException e) {e.printStackTrace();}System.out.println(getName()+"生产完成:"+bq);}}}

消费者类:

import java.util.concurrent.BlockingQueue;public class Consumer extends Thread{private BlockingQueue<String> bq;public Consumer(BlockingQueue<String> bq) {this.bq = bq;}public void run() {while(true) {System.out.println(getName()+"消费者准备消费集合元素!");try {Thread.sleep(200);//尝试取出元素,如果队列已空,则线程被阻bq.take();}catch(InterruptedException e) {e.printStackTrace();}System.out.println(getName()+"消费完成:"+bq);}}}

测试生产者消费者的测试类:

import java.util.concurrent.ArrayBlockingQueue;
import java.util.concurrent.BlockingQueue;public class BlockingQueueTest {public static void main(String[] args) {//创建一个容量为1的BlockingQueueBlockingQueue<String> bq = new ArrayBlockingQueue<String>(1);//启动三个生产者线程new Producer(bq).start();new Producer(bq).start();new Producer(bq).start();//启动一个消费者线程new Consumer(bq).start();}}

从上面程序可以看到,启动了3个生产者线程向BlockingQueue集合放入元素,启动了一个消费者线程从BlockingQueue集合取出元素。这段代码中BlockingQueue的容量为1,所以三个生产者线程不能连续放入元素,需要等消费者线程取出一个元素后,3个生产者线程的其中之一才能放入一个元素。运行结果如下:


(参考《疯狂Java讲义第3版》)

这篇关于多线程学习(八)——线程通信之阻塞队列(BlockingQueue)控制线程通信(生产者、消费者问题)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/238041

相关文章

Springboot的ThreadPoolTaskScheduler线程池轻松搞定15分钟不操作自动取消订单

《Springboot的ThreadPoolTaskScheduler线程池轻松搞定15分钟不操作自动取消订单》:本文主要介绍Springboot的ThreadPoolTaskScheduler线... 目录ThreadPoolTaskScheduler线程池实现15分钟不操作自动取消订单概要1,创建订单后

详谈redis跟数据库的数据同步问题

《详谈redis跟数据库的数据同步问题》文章讨论了在Redis和数据库数据一致性问题上的解决方案,主要比较了先更新Redis缓存再更新数据库和先更新数据库再更新Redis缓存两种方案,文章指出,删除R... 目录一、Redis 数据库数据一致性的解决方案1.1、更新Redis缓存、删除Redis缓存的区别二

oracle数据库索引失效的问题及解决

《oracle数据库索引失效的问题及解决》本文总结了在Oracle数据库中索引失效的一些常见场景,包括使用isnull、isnotnull、!=、、、函数处理、like前置%查询以及范围索引和等值索引... 目录oracle数据库索引失效问题场景环境索引失效情况及验证结论一结论二结论三结论四结论五总结ora

element-ui下拉输入框+resetFields无法回显的问题解决

《element-ui下拉输入框+resetFields无法回显的问题解决》本文主要介绍了在使用ElementUI的下拉输入框时,点击重置按钮后输入框无法回显数据的问题,具有一定的参考价值,感兴趣的... 目录描述原因问题重现解决方案方法一方法二总结描述第一次进入页面,不做任何操作,点击重置按钮,再进行下

解决mybatis-plus-boot-starter与mybatis-spring-boot-starter的错误问题

《解决mybatis-plus-boot-starter与mybatis-spring-boot-starter的错误问题》本文主要讲述了在使用MyBatis和MyBatis-Plus时遇到的绑定异常... 目录myBATis-plus-boot-starpythonter与mybatis-spring-b

C语言线程池的常见实现方式详解

《C语言线程池的常见实现方式详解》本文介绍了如何使用C语言实现一个基本的线程池,线程池的实现包括工作线程、任务队列、任务调度、线程池的初始化、任务添加、销毁等步骤,感兴趣的朋友跟随小编一起看看吧... 目录1. 线程池的基本结构2. 线程池的实现步骤3. 线程池的核心数据结构4. 线程池的详细实现4.1 初

mysql主从及遇到的问题解决

《mysql主从及遇到的问题解决》本文详细介绍了如何使用Docker配置MySQL主从复制,首先创建了两个文件夹并分别配置了`my.cnf`文件,通过执行脚本启动容器并配置好主从关系,文中还提到了一些... 目录mysql主从及遇到问题解决遇到的问题说明总结mysql主从及遇到问题解决1.基于mysql

如何测试计算机的内存是否存在问题? 判断电脑内存故障的多种方法

《如何测试计算机的内存是否存在问题?判断电脑内存故障的多种方法》内存是电脑中非常重要的组件之一,如果内存出现故障,可能会导致电脑出现各种问题,如蓝屏、死机、程序崩溃等,如何判断内存是否出现故障呢?下... 如果你的电脑是崩溃、冻结还是不稳定,那么它的内存可能有问题。要进行检查,你可以使用Windows 11

如何安装HWE内核? Ubuntu安装hwe内核解决硬件太新的问题

《如何安装HWE内核?Ubuntu安装hwe内核解决硬件太新的问题》今天的主角就是hwe内核(hardwareenablementkernel),一般安装的Ubuntu都是初始内核,不能很好地支... 对于追求系统稳定性,又想充分利用最新硬件特性的 Ubuntu 用户来说,HWEXBQgUbdlna(Har

MAVEN3.9.x中301问题及解决方法

《MAVEN3.9.x中301问题及解决方法》本文主要介绍了使用MAVEN3.9.x中301问题及解决方法,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面... 目录01、背景02、现象03、分析原因04、解决方案及验证05、结语本文主要是针对“构建加速”需求交