这些队列的 api ,就是添加队列,出队列,检测对首元素, 由于 add()--remove(), offer()--poll(),太简单这里不做验证, 只验证后二组api: 阻塞等待( put()--take() ), 超时等待( offer()--poll() )

多线程下,验证:

public static void main(String[] args) throws InterruptedException {
ArrayBlockingQueue<String> arrayBlockingQueue = new ArrayBlockingQueue(3);
/**
* 阻塞 等待 的 api: 添加队列put(), 出队列take()
*
* take(),从队列中取元素, 如果队列中没有元素,就在这里阻塞,直到队列中有元素,他取走为止, 不拿到元素死不罢休
* put(), 放元素到队列中, 如果队列已经满了,就在这里阻塞,直到队列的元素被取走,取走一个放入一个, 一定要放元素到队列中,哪怕等到天荒地老
*/
//验证 take(), 不拿到元素死不罢休
new Thread(()->{
try {
System.out.println(Thread.currentThread().getName() + "这里获取不到任何东西,所以在这里阻塞等待....");
String take = arrayBlockingQueue.take();
System.out.println(Thread.currentThread().getName() + "获取到的值为:" + take);
} catch (InterruptedException e) {
e.printStackTrace();
}
},"线程1").start(); // 这里睡1秒,目的是让 线程1 先执行
TimeUnit.SECONDS.sleep(1); new Thread(()->{
try {
System.out.println(Thread.currentThread().getName() + "抢到执行权, 此线程放数据开始");
arrayBlockingQueue.put("1");
arrayBlockingQueue.put("2");
arrayBlockingQueue.put("3");
System.out.println(Thread.currentThread().getName() + "放入成功");
} catch (InterruptedException e) {
e.printStackTrace();
}
},"线程2").start();
while(Thread.activeCount()>2){}
System.out.println("打印2:"+arrayBlockingQueue);
}

运行结果:

ArrayBlockingQueue<String> arrayBlockingQueue = new ArrayBlockingQueue(3);
/**
* 阻塞 等待 的 api: 添加队列put(), 出队列take()
*
* take(),从队列中取元素, 如果队列中没有元素,就在这里阻塞,直到队列中有元素,他取走为止, 不拿到元素死不罢休
* put(), 放元素到队列中, 如果队列已经满了,就在这里阻塞,直到队列的元素被取走,取走一个放入一个, 一定要放元素到队列中,哪怕等到天荒地老
*/
//验证 put(), 一定要放元素到队列,哪怕等到天荒地老
arrayBlockingQueue.put("123");
arrayBlockingQueue.put("456");
arrayBlockingQueue.put("789");
System.out.println("打印1:"+arrayBlockingQueue);
new Thread(()->{
try {
System.out.println(Thread.currentThread().getName() + "想放元素000 到队列中,但此时队列满了,所以开始等待");
arrayBlockingQueue.put("000");
System.out.println(Thread.currentThread().getName() + "等待结束了, 终于将000放入队列中了, 结束");
} catch (InterruptedException e) {
e.printStackTrace();
}
},"线程1").start(); // 这里睡1秒,目的是让 线程1 先执行
TimeUnit.SECONDS.sleep(1); new Thread(()->{
try {
System.out.println(Thread.currentThread().getName() + "抢到执行权. 开始移除一个元素");
String take = arrayBlockingQueue.take();
System.out.println(Thread.currentThread().getName() + "将 "+take+" 移除队列, 方便 线程1 放元素");
} catch (InterruptedException e) {
e.printStackTrace();
}
},"线程2").start(); while(Thread.activeCount()>2){}
System.out.println("打印2:"+arrayBlockingQueue);

运行结果:

超时等待 的 api 代码验证

    ArrayBlockingQueue<String> arrayBlockingQueue = new ArrayBlockingQueue(3);
/**
* 超时等待:带等待时间参数的: 添加队列 offer(,,) 移除队列 poll(,,)
*
* offer(,,) : 队列满了, 就在这里等待, 如果一定时间内.还没有元素被取出, 超过这个世界就退出等待
* poll(,,) : 队列为空, 就在这里等待, 如果一定时间内,还没有元素添加到队列中, 超过这个时间就退出等待
*/
//验证offer(,,)
arrayBlockingQueue.put("123");
arrayBlockingQueue.put("456");
arrayBlockingQueue.put("789");
System.out.println("初始队列:"+arrayBlockingQueue); new Thread(()->{
try {
System.out.println(Thread.currentThread().getName() + "想要往队列中放元素 0000, 但此时队列已满,等待3秒,3秒之内如果没有元素移除 , 就退出等待,不添加");
arrayBlockingQueue.offer("0000",3,TimeUnit.SECONDS);
} catch (InterruptedException e) {
e.printStackTrace();
}
},"线程1").start(); new Thread(()->{
try {
//TimeUnit.SECONDS.sleep(5);
System.out.println(Thread.currentThread().getName() + "抢到执行权,开始移除一个元素");
arrayBlockingQueue.poll();
} catch (Exception e) {
e.printStackTrace();
}
},"线程2").start(); while(Thread.activeCount()>2){}
System.out.println("打印2:"+arrayBlockingQueue);
}

运行结果为: 可以看到,队列满, 线程1等了3秒, 3秒内刚好线程2 移除了一个元素, 所以线程1,退出等待,添加0000到队列中

如果: 线程2 睡5秒呢? 此时可以看到, 线程1 等了3秒,队列依旧是满的, 就退出等待,不执行添加操作, 运行结果为:

验证 超时等待 poll(,,)

    ArrayBlockingQueue<String> arrayBlockingQueue = new ArrayBlockingQueue(3);
/**
* 超时等待:带等待时间参数的: 添加队列 offer(,,) 移除队列 poll(,,)
*
* offer(,,) : 队列满了, 就在这里等待, 如果一定时间内.还没有元素被取出, 超过这个世界就退出等待
* poll(,,) : 队列为空, 就在这里等待, 如果一定时间内,还没有元素添加到队列中, 超过这个时间就退出等待
*/
//验证 poll(,,)
new Thread(()->{
try {
System.out.println(Thread.currentThread().getName() + "此时队列为空, 等待3秒,如果3秒内,队列仍然为空,就退出等待,不为空就执行出队列操作");
String poll = arrayBlockingQueue.poll(3, TimeUnit.SECONDS);
System.out.println(Thread.currentThread().getName() + "取出的元素是:" + poll);
} catch (InterruptedException e) {
e.printStackTrace();
}
},"线程1").start(); new Thread(()->{
try {
//TimeUnit.SECONDS.sleep(5);
System.out.println(Thread.currentThread().getName() + "开始往队列中放元素");
arrayBlockingQueue.offer("123");
arrayBlockingQueue.offer("456");
System.out.println(Thread.currentThread().getName() + "入队结束");
} catch (Exception e) {
e.printStackTrace();
}
},"线程2").start();
while(Thread.activeCount()>2){}
System.out.println("打印2:"+arrayBlockingQueue);

运行结果: 队列空,线程1,等待3秒, 刚好3秒内,线程2,往队列中添加了元素, 所以 线程1 取到了元素



如果线程2 等待5秒: 线程1将取不到元素 为null, 结果为:

JUC 并发编程--06, 阻塞队列(7种), 阻塞等待 api的 代码验证的更多相关文章

  1. JUC 并发编程--12, 使用AtomicInteger 实现一把锁(排队自旋锁), 代码演示

    前面 使用自旋锁实现了一把锁,(请看 第5篇) volatile 三大特性: 可见性, 不保证原子性, 禁止指令重排 为了解决 volatile不保证原子性的问题, 引入了原子类, AtomicInt ...

  2. 并发编程学习笔记(13)----ConcurrentLinkedQueue(非阻塞队列)和BlockingQueue(阻塞队列)原理

    · 在并发编程中,我们有时候会需要使用到线程安全的队列,而在Java中如果我们需要实现队列可以有两种方式,一种是阻塞式队列.另一种是非阻塞式的队列,阻塞式队列采用锁来实现,而非阻塞式队列则是采用cas ...

  3. Python并发编程06 /阻塞、异步调用/同步调用、异步回调函数、线程queue、事件event、协程

    Python并发编程06 /阻塞.异步调用/同步调用.异步回调函数.线程queue.事件event.协程 目录 Python并发编程06 /阻塞.异步调用/同步调用.异步回调函数.线程queue.事件 ...

  4. 并发编程 06—— CompletionService :Executor 和 BlockingQueue

    Java并发编程实践 目录 并发编程 01—— ThreadLocal 并发编程 02—— ConcurrentHashMap 并发编程 03—— 阻塞队列和生产者-消费者模式 并发编程 04—— 闭 ...

  5. JUC并发编程学习笔记

    JUC并发编程学习笔记 狂神JUC并发编程 总的来说还可以,学到一些新知识,但很多是学过的了,深入的部分不多. 线程与进程 进程:一个程序,程序的集合,比如一个音乐播发器,QQ程序等.一个进程往往包含 ...

  6. Python 3 并发编程多进程之队列(推荐使用)

    Python 3 并发编程多进程之队列(推荐使用) 进程彼此之间互相隔离,要实现进程间通信(IPC),multiprocessing模块支持两种形式:队列和管道,这两种方式都是使用消息传递的. 可以往 ...

  7. 自己总结 :并发队列ConcurrentLinkedQueue、阻塞队列AraayBlockingQueue、阻塞队列LinkedBlockingQueue 区别 和 使用场景总结

    并发队列ConcurrentLinkedQueue.阻塞队列AraayBlockingQueue.阻塞队列LinkedBlockingQueue 区别 和  使用场景总结 分类: Java2013-0 ...

  8. JUC并发编程基石AQS之主流程源码解析

    前言 由于AQS的源码太过凝练,而且有很多分支比如取消排队.等待条件等,如果把所有的分支在一篇文章的写完可能会看懵,所以这篇文章主要是从正常流程先走一遍,重点不在取消排队等分支,之后会专门写一篇取消排 ...

  9. JUC 并发编程--07 阻塞队列版本的 生产者消费者(不使用synchronized和 lock),也有一些疑惑,最终解惑

    直接上代码: 前提是你已经 熟悉了原子类,volatile,和阻塞队列 public class JucPCdemo03 { /** * 阻塞队列的应用: 这里实现的生产者消费者,生产一个消费一个 * ...

随机推荐

  1. Winamp栈溢出漏洞研究【转载】

    课程简介 Winamp是一款非常经典的音乐播放软件,它于上世纪九十年代后期问世.与现在音乐播放软件行业百家争鸣的情况不同,当时可以说Winamp就是听音乐的唯一选择了,相信那个时代的电脑玩家是深有体会 ...

  2. windows核心编程-第一章 对程序错误的处理

    第一章-对程序错误的处理 在开始介绍Microsoft Windows 的特性之前,必须首先了解 Wi n d o w s的各个函数是如何进行错误处理的. 当调用一个Wi n d o w s函数时,它 ...

  3. 深入浅出带你玩转sqlilabs(三)--GET,POST,COOKIE,万能密码注入

    常见提交方式下的注入漏洞 WEB应用在数据传递接受中,针对SQL注入安全漏洞,由于数据大小,格式等原因,脚本在接受传递时会有多种传递方式,传递方式的不同将影响到安全测试的不同 第一点:数据常见提交方式 ...

  4. 【HTML】HTML5从入门到深入(复习查漏向

    HTML5从入门到深入(复习查漏向 冷知识 万维网之始:第一个网站·蒂姆伯纳斯-李 HTML5 html5文档类型声明: 头部: <!doctype html> 字符集(charset): ...

  5. Day005 for循环

    for循环 虽然所有循环结构都可以用while或者do-while表示,但java提供了另一种语句--for循环,使一些循环结构变得更加简单. for循环语句是支持迭代的一种通用结构,是最有效.最灵活 ...

  6. surging 基于流媒体服务如何集群分流

    前言 最近几年微服务可谓是大火,大家忙着建设微服务,学习微服务如何搭建,微服务技术体系的演变也使得企业公司能支持起灵活,多样化的业务需求和越来越多的访问量,有很多企业用户正在朝着业务中台,SAAS云平 ...

  7. 【vue-07】vue-router

    Vue-router官网 安装 vue-router是一个插件包,所以我们还是需要用npm 来进行安装.打开命令行工具,进入你的项目目录,输入下面命令. npm install vue-router ...

  8. 逆向工程初步160个crackme-------6

    工具:1. 按钮事件地址转换器E2A 2. PEID 3. Ollydbg 同样我们先来运行一下这个程序, ok按钮是被禁用的,有一个help按钮点击后弹出一个消息框:消息框显示提示信息为.本程序需要 ...

  9. Pytorch_Part7_模型使用

    VisualPytorch beta发布了! 功能概述:通过可视化拖拽网络层方式搭建模型,可选择不同数据集.损失函数.优化器生成可运行pytorch代码 扩展功能:1. 模型搭建支持模块的嵌套:2. ...

  10. laravel 批量删除

    <button id="delAll">批量删除</button>//给按钮一个id属性 <input type="checkbox&quo ...