生产者消费者问题是一个著名的线程同步问题,该问题描述如下:有一个生产者在生产产品,这些产品将提供给若干个消费者去消费,为了使生产者和消费者能并发执行,在两者之间设置一个具有多个缓冲区的缓冲池,生产者将它生产的产品放入一个缓冲区中,消费者可以从缓冲区中取走产品进行消费,显然生产者和消费者之间必须保持同步,即不允许消费者到一个空的缓冲区中取产品,也不允许生产者向一个已经放入产品的缓冲区中再次投放产品。

使用synchronized关键字实现线程同步

在使用wait()和notifyAll()方法时,应注意将wait()方法放入循环中,否则会产生虚假唤醒问题。

/**
* Created by 吴海飞 on 2017-1-23.
*/
public class TestProductAndConsumer {
public static void main(String[] args){
Clerk clerk = new Clerk();
Productor pro = new Productor(clerk);
Consumer consumer = new Consumer(clerk);
new Thread(pro,"生产者A").start();
new Thread(consumer,"消费者B").start();
}
} /**
* 店员,可以进货与销售货物
*/
class Clerk{ private int product = 0; /**
* 进货的方法
*/
public synchronized void get(){
while (product>=1){//为了避免虚假唤醒问题,应该总是使用在循环中
System.out.println("产品已满!"); try {
this.wait();//等待
} catch (InterruptedException e) {
e.printStackTrace();
}
}
System.out.println(Thread.currentThread().getName() + ":" + ++product);
this.notifyAll();//唤醒线程
} /**
* 销售的方法
*/
public synchronized void sale(){
while (product<=0){//为避免虚假唤醒,应该总是始终使用在循环中
System.out.println("缺货……");
try {
this.wait();
} catch (InterruptedException e) {
e.printStackTrace();
}
}
System.out.println(Thread.currentThread().getName() + ":"+ --product);
this.notifyAll();
}
} /**
* 生产者
*/
class Productor implements Runnable{ private Clerk clerk; public Productor(Clerk clerk){
this.clerk = clerk;
} @Override
public void run() {
for (int i = 0; i < 10; i++){
try {
Thread.sleep(200);
} catch (InterruptedException e) {
e.printStackTrace();
}
clerk.get();
}
}
} /**
* 消费者
*/
class Consumer implements Runnable{ private Clerk clerk; public Consumer(Clerk clerk){
this.clerk = clerk;
} @Override
public void run() {
for (int i = 0; i < 10; i++){
clerk.sale();
}
}
}

使用同步锁实现线程同步问题

使用同步锁时应注意lock()与unlock()方法的同步使用。


import java.util.concurrent.locks.Condition;
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantLock; /**
* 使用ReentrantLock实现生产者消费者问题
* Created by 吴海飞 on 2017-1-23.
*/
public class TestReentrantLock {
public static void main(String[] args){
Clerk clerk = new Clerk();
Productor pro = new Productor(clerk);
Consumer consumer = new Consumer(clerk); new Thread(pro,"生产者A").start();
new Thread(consumer,"消费者B").start();
new Thread(pro,"生产者C").start();
new Thread(consumer,"消费者D").start();
}
} class Clerk{
private Lock lock = new ReentrantLock();//获取同步锁
private Condition condition = lock.newCondition();
private int product = 0; /**
* 进货的方法
*/ public void get(){
lock.lock();//打开锁
try{
while (product>=1){//为了避免虚假唤醒问题,应该总是使用在循环中
System.out.println("产品已满!");
try {
condition.await();
} catch (InterruptedException e) {
e.printStackTrace();
}
}
System.out.println(Thread.currentThread().getName() + ":" + ++product);
condition.signalAll();
}finally {
lock.unlock();//关闭锁
} } /**
* 销售的方法
*/ public void sale(){
lock.lock();//加锁
try {
while (product<=0){//为避免虚假唤醒,应该总是始终使用在循环中
System.out.println("缺货……");
try {
condition.await();//等待
} catch (InterruptedException e) {
e.printStackTrace();
}
}
System.out.println(Thread.currentThread().getName() + ":"+ --product);
condition.signalAll();//唤醒等待
}finally {
lock.unlock();//释放锁
} }
} /**
* 生产者
*/ class Productor implements Runnable{ private Clerk clerk; public Productor(Clerk clerk){
this.clerk = clerk;
} @Override
public void run() {
for (int i = 0; i < 10; i++){
try {
Thread.sleep(200);
} catch (InterruptedException e) {
e.printStackTrace();
}
clerk.get();
}
}
} /**
* 消费者
*/ class Consumer implements Runnable{ private Clerk clerk; public Consumer(Clerk clerk){
this.clerk = clerk;
} @Override
public void run() {
for (int i = 0; i < 10; i++){
clerk.sale();
}
}
}

最新文章

  1. 李洪强iOS经典面试题142-第三方框架及其管理
  2. Weblogic的安装与配置
  3. Adaboost算法初识
  4. 他们在军训,我在搞 OI(一)
  5. SQL Server 字符串处理
  6. POJ 2516 最小费用最大流
  7. Modem常用概念
  8. Ubutu命令 笔记积累
  9. magento后台登陆404、Front controller reached 100 router match iterations的解决方案
  10. ftk学习记录(一个进度条文章)
  11. shell 命令合并文本
  12. 芝麻HTTP:TensorFlow LSTM MNIST分类
  13. 10_Android中通过HttpUrlConnection访问网络,Handler和多线程使用,读取网络html代码并显示在界面上,ScrollView组件的使用
  14. 爬虫基础之urllib库(代码演示)
  15. js调起微信客户端
  16. Docker简单使用
  17. 使用fastjson解析数据后导致顺序改变问题
  18. Linux&#160;配置iso系统盘为本地yum源
  19. jquery日期和时间的插件精确到秒
  20. DA-GAN技术【简介】【机器通过文字描述创造图像】

热门文章

  1. Azure VNet介绍
  2. QT5 地图的使用
  3. Java-API:java.util.map
  4. C Primer Plus学习笔记(六)- C 控制语句:分支和跳转
  5. jdbcTemplate学习(四)
  6. hadoop 常用端口 及模块介绍
  7. AndroidStudio 中使用FFMPEG
  8. Anaconda 安装教程(Win10环境) Tensorflow安装
  9. ActionbarActivity和普通的Activity的区别
  10. zabbix监控报错zabbix server is not running: the information displayed may not be current