JUC学习之生产者和消费者
时间:2022-07-22
本文章向大家介绍JUC学习之生产者和消费者,主要内容包括其使用实例、应用技巧、基本知识点总结和需要注意事项,具有一定的参考价值,需要的朋友可以参考一下。
生产者和消费者问题是线程模型中的经典问题:生产者和消费者在同一时间段内共用同一个存储空间,生产者往存储空间中添加产品,消费者从存储空间中取走产品,当存储空间为空时,消费者阻塞,当存储空间满时,生产者阻塞。我们这里用信号灯的变化来模拟演示。
synchronized版:
wait()
和 notify()
方法的实现,这也是最简单最基础的实现,缓冲区满和为空时都调用wait()方法等待,当生产者生产了一个产品或者消费者消费了一个产品之后会唤醒所有线程。
/**
* 信号灯
* 红绿两种状态
* 一个改变颜色的方法
*/
class TrafficLight{
private String color = "red";
/**
* 是,就等待
* 不是,就执行
* 然后,唤醒
*/
public synchronized void change2Green() throws InterruptedException {
while (color.equals("green")) {
this.wait();
}
// TimeUnit.SECONDS.sleep(1);
color = "green";
System.out.println(Thread.currentThread().getName()+"==>"+color);
this.notifyAll();
}
public synchronized void change2Red() throws InterruptedException {
while (color.equals("red")) {
this.wait();
}
color = "red";
System.out.println(Thread.currentThread().getName()+"==>"+color);
this.notifyAll();
}
}
需要注意的是:当使用if
判断时,会造成虚假唤醒
,解决方法为:
Lock版:
可重入锁ReentrantLock
的实现,java.util.concurrent.lock 中的 Lock 框架是锁定的一个抽象,通过对lock的lock()
方法和unlock()
方法实现了对锁的显示控制,而synchronize()则是对锁的隐性控制。
可重入锁
,也叫做递归锁,指的是同一线程 外层函数获得锁之后 ,内层递归函数仍然有获取该锁的代码,但不受影响,简单来说,该锁维护这一个与获取锁相关的计数器,如果拥有锁的某个线程再次得到锁,那么获取计数器就执行,函数调用结束计数器就执行,然后锁需要被释放两次才能获得真正释放。已经获取锁的线程进入其他需要相同锁的同步代码块不会被阻塞。
class TrafficLight2 {
private String color = "red";
/**
* lock版本的:await()=wait(),signal()=notify()
* Condition可以单独监视一个线程,实现精准通知唤醒
*/
Lock lock = new ReentrantLock();
Condition condition1 = lock.newCondition();
Condition condition2 = lock.newCondition();
Condition condition3 = lock.newCondition();
public void change2Green() throws InterruptedException {
lock.lock();
try {
while (color.equals("green")) {
condition1.await();
}
// TimeUnit.SECONDS.sleep(1);
color = "green";
System.out.println(Thread.currentThread().getName() + "==>" + color);
condition3.signal();
} catch (InterruptedException e) {
e.printStackTrace();
} finally {
lock.unlock();
}
}
public void change2Red() throws InterruptedException {
lock.lock();
try {
while (color.equals("red")) {
condition2.await();
}
color = "red";
System.out.println(Thread.currentThread().getName() + "==>" + color);
condition1.signal();
} catch (InterruptedException e) {
e.printStackTrace();
} finally {
lock.unlock();
}
}
public void change2Yellow() throws InterruptedException {
lock.lock();
try {
while (color.equals("yellow")) {
condition3.await();
}
color = "yellow";
System.out.println(Thread.currentThread().getName() + "==>" + color);
condition2.signal();
} catch (InterruptedException e) {
e.printStackTrace();
} finally {
lock.unlock();
}
}
}
await()= wait(),signal()= notify()
Condition可以单独监视一个线程,实现精准通知唤醒
Q.E.D.
- JavaScript 教程
- JavaScript 编辑工具
- JavaScript 与HTML
- JavaScript 与Java
- JavaScript 数据结构
- JavaScript 基本数据类型
- JavaScript 特殊数据类型
- JavaScript 运算符
- JavaScript typeof 运算符
- JavaScript 表达式
- JavaScript 类型转换
- JavaScript 基本语法
- JavaScript 注释
- Javascript 基本处理流程
- Javascript 选择结构
- Javascript if 语句
- Javascript if 语句的嵌套
- Javascript switch 语句
- Javascript 循环结构
- Javascript 循环结构实例
- Javascript 跳转语句
- Javascript 控制语句总结
- Javascript 函数介绍
- Javascript 函数的定义
- Javascript 函数调用
- Javascript 几种特殊的函数
- JavaScript 内置函数简介
- Javascript eval() 函数
- Javascript isFinite() 函数
- Javascript isNaN() 函数
- parseInt() 与 parseFloat()
- escape() 与 unescape()
- Javascript 字符串介绍
- Javascript length属性
- javascript 字符串函数
- Javascript 日期对象简介
- Javascript 日期对象用途
- Date 对象属性和方法
- Javascript 数组是什么
- Javascript 创建数组
- Javascript 数组赋值与取值
- Javascript 数组属性和方法
- Yii框架的路由配置方法分析
- Android 购物车加减功能的实现代码
- Yii框架函数简单用法分析
- PHP读取XML文件的方法实例总结【DOMDocument及simplexml方法】
- 浅析PHP7 的垃圾回收机制
- android BottomSheetDialog新控件解析实现知乎评论列表效果(实例代码)
- Laravel框架实现的上传图片到七牛功能详解
- Flutter沉浸式状态栏/AppBar导航栏/仿咸鱼底部凸起导航栏效果
- Android集成zxing扫码框架功能
- Android 实现抖音小游戏潜艇大挑战的思路详解
- 修改Android Studio 的 Logcat 缓冲区大小操作
- Android自定义View验证码输入框
- PHP生成随机字符串实例代码(字母+数字)
- Laravel框架中缓存的使用方法分析
- android实现搜索功能并将搜索结果保存到SQLite中(实例代码)