Condition对象

一)、Condition的定义

Condition对象:与锁关联,协调多线程间的复杂协作。

获取与锁绑定的Condition对象:

Lock lock = new ReentrantLock();

Conndition condition = lock.newConndition();

Condition的方法:

await(): 使当前的线程等待并释放锁。

singalAll(): 唤醒所有等待的线程,只有一个线程重新获得锁,并执行。

awaitterruptibly(): 使当前线程等待并释放锁,但在等待过程中不响应中断。

singal(): 唤醒一个正在等待的线程。

注:当线程处于中断状态也能跳出等待。

Condition对象与锁的关系

相当于object.wait(),Object.notify()与Synchronized一样,配合使用,以完成多线程协作的控制。

Lock lock = new ReentrantLock();
Conndition condition = lock.newCondition();
//线程进入等待状态
condition.await();
//唤醒等待的线程
condition.notify();

二)、ArrayBlockingQueue: 使用Condition实现队列的阻塞

(2-1):ArrayBlockQueue的主要属性

 //存放元素的数组,当元素个数超过数组的长度,调用的线程进入阻塞状态
final Object[] items;
//取出元素的数组下标
int takeIndex;
//添加元素的下标
int putIndex;
//数组中元素的个数
int count;
//公共锁对象
final ReentrantLock lock;
//与锁相对应的Condition(锁的监视器)
//等待队列不为空的时候
private final Condition notEmpty;
//等待队列不为满的时候
private final Condition notFull;

(2-2):构造方法

    public ArrayBlockingQueue(int capacity, boolean fair) {
if (capacity <= 0)
throw new IllegalArgumentException();
this.items = new Object[capacity];
//创建公共锁对象,final修饰,对象一创建不能修改,final修饰的变量不能改变他的引用地址,但是可以改变它的值。
lock = new ReentrantLock(fair);
//与锁相关联的Condition
notEmpty = lock.newCondition();
notFull = lock.newCondition();
}

(2-3):ArrayBlockingQueue的主要方法:

1). put(E e): 添加元素到队列

特点:

1).当队列的元素满时,阻塞当前的线程。

​ noFull.await() : 等待队列不满的时候。

2).添加一个元素后,唤醒一个消费线程。

​ noEmpty.singnal(): 队列不空,发出信号,唤醒等待不为空的线程

    public void put(E e) throws InterruptedException {
checkNotNull(e);
final ReentrantLock lock = this.lock;
lock.lockInterruptibly();
try {
while (count == items.length)
//等待队列不为满的时候
notFull.await();
enqueue(e);
} finally {
lock.unlock();
}
} //入队
private void enqueue(E x) {
// assert lock.getHoldCount() == 1;
// assert items[putIndex] == null;
final Object[] items = this.items;
items[putIndex] = x;
if (++putIndex == items.length)
putIndex = 0;
count++;
//唤醒等待不为空的线程
notEmpty.signal();
}

2). take(): 获取队列中的元素

特点:

1).当队列为空时,阻塞当前调用线程。

​ noEmpty.await: 等待队列非空的时候

2).获取一个元素,唤醒一个生产线程。

​ noFull.singnal(): 唤醒一个等待队列不为满的线程

    public E take() throws InterruptedException {
final ReentrantLock lock = this.lock;
lock.lockInterruptibly();
try {
while (count == 0)
//线程进入等待状态,等待队列非空
notEmpty.await();
return dequeue();
} finally {
lock.unlock();
}
} //出列
private E dequeue() {
// assert lock.getHoldCount() == 1;
// assert items[takeIndex] != null;
final Object[] items = this.items;
@SuppressWarnings("unchecked")
E x = (E) items[takeIndex];
items[takeIndex] = null;
if (++takeIndex == items.length)
takeIndex = 0;
count--;
if (itrs != null)
itrs.elementDequeued();
//唤醒一个等待队列不为满时的线
notFull.signal();
return x;
}

(2-4):队列先进先出的控制

putIndex: 0 - item.length, count++

if (++putIndex == items.length)
putIndex = 0;
count++;

takeIndex:0 - item.length,count--

if (++takeIndex == items.length)
//控制着元素的先进先出
takeIndex = 0;
count--;

总结: 元素从0开始写入,从0开始读取,由count控制着元素的读取,当putIndex =

​ item.length时,只要count不等于item.length,那么item[0]的元素必定被消

​ 费,当takeIndex = item.length时,只要count不等于0,item[0]必有元素存

​ 在。

count: 0 - item.length,控制着是否读取元素或写入元素

三)、使用ArrayBlockingQueue来构建生产者 - 消费者模式

生产者:Producer

/**
* 生产者进程
*
* 生产者 - 消费者模式
* 共同维护一个存储队列
* 队列特点:
* 队列满时,阻塞生产者,线程进入等待状态。
* 队列为空时,阻塞消费者,线程进入等待状态。
*/
public class Producer extends Thread {
/**
* 生产者维护的生产队列,指明队列存储元素的大小
*/
protected ArrayBlockingQueue queue; public Producer(ArrayBlockingQueue queue,String name) {
super(name);
this.queue = queue;
} /**
* 生产者生产线程
*/
@Override
public void run() {
try {
Thread.sleep(100);
Object obj = new Object();
System.out.println(Thread.currentThread().getName()+" 正在生产");
//模拟生产者的生产过程
queue.put(obj);
System.out.println(Thread.currentThread().getName()+" 生产了一件商品。。。");
} catch (InterruptedException e) {
e.printStackTrace();
}
} }

消费者:Consumer

/**
* 消费者线程,维护了一个消费队列
*/
public class Consumer extends Thread{
protected ArrayBlockingQueue queue; public Consumer(ArrayBlockingQueue queue,String name) {
super(name);
this.queue = queue;
} @Override
public void run() {
//模拟消费者线程进行消费
try {
System.out.println(Thread.currentThread().getName()+" 要开始消费了");
queue.take();
System.out.println(Thread.currentThread().getName()+" 消费了一件商品");
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}

启动线程:Client

/**
* 开启多个线程进行生产,开启多个线程进行消费
*/ public class Client {
public static void main(String[] args) {
ArrayBlockingQueue queue = new ArrayBlockingQueue(20);
ExecutorService executor = Executors.newFixedThreadPool(200);
Producer producer1 = new Producer(queue,"producer - 1");
Producer producer2 = new Producer(queue,"producer - 2");
Producer producer3 = new Producer(queue,"producer - 3");
Consumer consumer1 = new Consumer(queue,"consumer - 1");
Consumer consumer2 = new Consumer(queue,"consumer - 2");
Consumer consumer3 = new Consumer(queue,"consumer - 3");
//开启多个线程进行生产
executor.execute(producer1);
executor.execute(producer2);
executor.execute(producer3); //开启多个线程进行消费
executor.execute(consumer1);
executor.execute(consumer2);
executor.execute(consumer3); }
}

结果:

pool-1-thread-5 要开始消费了
pool-1-thread-4 要开始消费了
pool-1-thread-6 要开始消费了
pool-1-thread-1 正在生产
pool-1-thread-1 生产了一件商品。。。
pool-1-thread-2 正在生产
pool-1-thread-3 正在生产
pool-1-thread-5 消费了一件商品
pool-1-thread-6 消费了一件商品
pool-1-thread-4 消费了一件商品
pool-1-thread-2 生产了一件商品。。。
pool-1-thread-3 生产了一件商品。。。

结果分析:

消费者消费ArrayBlockingQueue的数据时,当队列为空的时候会阻塞当前的线程,当生产者生产了一件商品后会唤醒一个阻塞的线程。

练习代码github地址:https://github.com/slob-cow/java_performance_optimization/tree/master/Condition

Condition对象以及ArrayBlockingQueue阻塞队列的实现(使用Condition在队满时让生产者线程等待, 在队空时让消费者线程等待)的更多相关文章

  1. ArrayBlockingQueue 阻塞队列和 Semaphore 信号灯的应用

    import java.util.concurrent.ArrayBlockingQueue; import java.util.concurrent.BlockingQueue; public cl ...

  2. ArrayBlockingQueue 阻塞队列 生产者 与消费者案例

    package com.originalityTest; import java.net.UnknownHostException; import java.util.ArrayList; impor ...

  3. java多线程8:阻塞队列与Fork/Join框架

    队列(Queue),是一种数据结构.除了优先级队列和LIFO队列外,队列都是以FIFO(先进先出)的方式对各个元素进行排序的. BlockingQueue 而阻塞队列BlockingQueue除了继承 ...

  4. 10.并发包阻塞队列之ArrayBlockingQueue

    上一节中对并发包中的非阻塞队列ConcurrentLinkedQueue的入队.出队做了一个简要的分析,本文将对并发包中的阻塞队列做一个简要分析. Java并发包中的阻塞队列一共7个,当然他们都是线程 ...

  5. 并发包阻塞队列之ArrayBlockingQueue

    并发包阻塞队列之ArrayBlockingQueue   jdk1.7.0_79  上一节中对并发包中的非阻塞队列ConcurrentLinkedQueue的入队.出队做了一个简要的分析,本文将对并发 ...

  6. 阻塞队列BlockingQueue之ArrayBlockingQueue

    ArrayBlockingQueue  是数组实现的有界阻塞队列,此队列按照先进先出(FIFO)的原则对元素进行排序. 构造方法: public ArrayBlockingQueue(int capa ...

  7. Java中的阻塞队列

    1. 什么是阻塞队列? 阻塞队列(BlockingQueue)是一个支持两个附加操作的队列.这两个附加的操作是:在队列为空时,获取元素的线程会等待队列变为非空.当队列满时,存储元素的线程会等待队列可用 ...

  8. Java并发编程之阻塞队列

    1.什么是阻塞队列? 队列是一种数据结构,它有两个基本操作:在队列尾部加入一个元素,从队列头部移除一个元素.阻塞队里与普通的队列的区别在于,普通队列不会对当前线程产生阻塞,在面对类似消费者-生产者模型 ...

  9. 聊聊并发(七)——Java中的阻塞队列

    3. 阻塞队列的实现原理 聊聊并发(七)--Java中的阻塞队列 作者 方腾飞 发布于 2013年12月18日 | ArchSummit全球架构师峰会(北京站)2016年12月02-03日举办,了解更 ...

随机推荐

  1. IDE安装完成无法打开,报错Fail load JVM DLL 问题与解决方案

    安装完成pycharm 在打开pycharm的时候出现报错 Fail load JVM DLL xxxx xxx. 解决方案 安装Microsoft Visual C++ 2010 Redistrib ...

  2. 详解 Redis 内存管理机制和实现

    Redis是一个基于内存的键值数据库,其内存管理是非常重要的.本文内存管理的内容包括:过期键的懒性删除和过期删除以及内存溢出控制策略. 最大内存限制 Redis使用 maxmemory 参数限制最大可 ...

  3. 百万年薪python之路 -- MySQL数据库之 Navicat工具和pymysql模块

    一. IDE工具介绍(Navicat) 生产环境还是推荐使用mysql命令行,但为了方便我们测试,可以使用IDE工具,我们使用Navicat工具,这个工具本质上就是一个socket客户端,可视化的连接 ...

  4. macport命令--笔记

    macport命令:sudo port sync //同步本地和全球的ports tree,但不检查自己是否有更新.sudo port install python36 //安装python36sud ...

  5. SpringBoot整合Swagger2(Demo示例)

    写在前面 由于公司项目采用前后端分离,维护接口文档基本上是必不可少的工作.一个理想的状态是设计好后,接口文档发给前端和后端,大伙按照既定的规则各自开发,开发好了对接上了就可以上线了.当然这是一种非常理 ...

  6. Dubbo与Kubernetes集成

    Dubbo应用迁移到docker的问题 Dubbo是阿里开源的一套服务治理与rpc框架,服务的提供者通过zookeeper把自己的服务发布上去,然后服务调用方通过zk获取服务的ip和端口,dubbo客 ...

  7. const var let 三者的区别

    1.const定义的变量不可以修改,而且必须初始化. ;//正确 const b;//错误,必须初始化 console.log('函数外const定义b:' + b);//有输出值 b = ; con ...

  8. mock和axios常见的传参方式

    第一次接手项目,传参方式还有些吃力,因此做一下总结. 首先我们需要会看swagger中的接口.里面写了某个接口需要接收什么样的值,前端怎么传递这个值 在mock中的传参方式: mock中传参的方式有两 ...

  9. Python实战练习——打印日历教程

    很长一段时间没有接触过C语言了,想来做这一行当已经有三两年了. 今天突然想起来以前用C语言在VC6上写代码的日子,想了想以前的一些实战练习. 所以今天打算用Python来写一个C语言以前练习的题目-日 ...

  10. 使用VM虚拟机安装Linux系统详细流程

    最近新换了个电脑,所以需要重新安装虚拟机和Linux系统,话不多说,看流程吧 1.安装vm,这个就不说了,打开VM 2.点击安装虚拟机 3.选择自定义安装 4.选择稍后安装 5.选择要安装的系统 6. ...