`

Java线程(三):线程协作-生产者/消费者模式

阅读更多

 

   上一篇讲述了线程的互斥(同步),但是在很多情况下,仅仅同步是不够的,还需要线程与线程协作(通信),生产者/消费者模式是一个经典的线程同步以及通信的模型。

        假设有这样一种情况,有一个盘子,盘子里只能放一个鸡蛋,A线程专门往盘子里放鸡蛋,如果盘子里有鸡蛋,则一直等到盘子里没鸡蛋,B线程专门从盘子里取鸡蛋,如果盘子里没鸡蛋,则一直等到盘子里有鸡蛋。这里盘子是一个互斥区,每次放鸡蛋是互斥的,每次取鸡蛋也是互斥的,A线程放鸡蛋,如果这时B线程要取鸡蛋,由于A没有释放锁,B线程处于等待状态,进入阻塞队列,放鸡蛋之后,要通知B线程取鸡蛋,B线程进入就绪队列,反过来,B线程取鸡蛋,如果A线程要放鸡蛋,由于B线程没有释放锁,A线程处于等待状态,进入阻塞队列,取鸡蛋之后,要通知A线程放鸡蛋,A线程进入就绪队列。我们希望当盘子里有鸡蛋时,A线程阻塞,B线程就绪,盘子里没鸡蛋时,A线程就绪,B线程阻塞,代码如下:

 

  1. import java.util.ArrayList;  
  2. import java.util.List;  
  3. /** 定义一个盘子类,可以放鸡蛋和取鸡蛋 */  
  4. public class Plate {  
  5.     /** 装鸡蛋的盘子 */  
  6.     List<Object> eggs = new ArrayList<Object>();  
  7.     /** 取鸡蛋 */  
  8.     public synchronized Object getEgg() {  
  9.         while (eggs.size() == 0) {  
  10.             try {  
  11.                 wait();  
  12.             } catch (InterruptedException e) {  
  13.                 e.printStackTrace();  
  14.             }  
  15.         }  
  16.         Object egg = eggs.get(0);  
  17.         eggs.clear();// 清空盘子  
  18.         notify();// 唤醒阻塞队列的某线程到就绪队列  
  19.         System.out.println("拿到鸡蛋");  
  20.         return egg;  
  21.     }  
  22.     /** 放鸡蛋 */  
  23.     public synchronized void putEgg(Object egg) {  
  24.         while (eggs.size() > 0) {  
  25.             try {  
  26.                 wait();  
  27.             } catch (InterruptedException e) {  
  28.                 e.printStackTrace();  
  29.             }  
  30.         }  
  31.         eggs.add(egg);// 往盘子里放鸡蛋  
  32.         notify();// 唤醒阻塞队列的某线程到就绪队列  
  33.         System.out.println("放入鸡蛋");  
  34.     }  
  35.     static class AddThread extends Thread {  
  36.         private Plate plate;  
  37.         private Object egg = new Object();  
  38.         public AddThread(Plate plate) {  
  39.             this.plate = plate;  
  40.         }  
  41.         public void run() {  
  42.             plate.putEgg(egg);  
  43.         }  
  44.     }  
  45.     static class GetThread extends Thread {  
  46.         private Plate plate;  
  47.         public GetThread(Plate plate) {  
  48.             this.plate = plate;  
  49.         }  
  50.         public void run() {  
  51.             plate.getEgg();  
  52.         }  
  53.     }  
  54.     public static void main(String args[]) {  
  55.         Plate plate = new Plate();  
  56.         for(int i = 0; i < 10; i++) {  
  57.             new Thread(new AddThread(plate)).start();  
  58.             new Thread(new GetThread(plate)).start();  
  59.         }  
  60.     }  
  61. }  

        输出结果:

 

 

  1. 放入鸡蛋  
  2. 拿到鸡蛋  
  3. 放入鸡蛋  
  4. 拿到鸡蛋  
  5. 放入鸡蛋  
  6. 拿到鸡蛋  
  7. 放入鸡蛋  
  8. 拿到鸡蛋  
  9. 放入鸡蛋  
  10. 拿到鸡蛋  
  11. 放入鸡蛋  
  12. 拿到鸡蛋  
  13. 放入鸡蛋  
  14. 拿到鸡蛋  
  15. 放入鸡蛋  
  16. 拿到鸡蛋  
  17. 放入鸡蛋  
  18. 拿到鸡蛋  
  19. 放入鸡蛋  
  20. 拿到鸡蛋  

        8 l程序开始,A线程判断盘子是否为空,放入一个鸡蛋,并且唤醒在阻塞队列的一个线程,阻塞队列为空;假设CPU又调度了一个A线程,盘子非空,执行等待,这个A线程进入阻塞队列;然后一个B线程执行,盘子非空,取走鸡蛋,并唤醒阻塞队列的A线程,A线程进入就绪队列,此时就绪队列就一个A线程,马上执行,放入鸡蛋;如果再来A线程重复第一步,在来B线程重复第二步,整个过程就是生产者(A线程)生产鸡蛋,消费者(B线程)消费鸡蛋。

 

        前段时间看了张孝祥老师线程的视频,讲述了一个其学员的面试题,也是线程通信的,在此也分享一下。

        题目:子线程循环10次,主线程循环100次,如此循环100次,好像是空中网的笔试题。

 

  1. public class ThreadTest2 {  
  2.     public static void main(String[] args) {  
  3.         final Business business = new Business();  
  4.         new Thread(new Runnable() {  
  5.             @Override  
  6.             public void run() {  
  7.                 threadExecute(business, "sub");  
  8.             }  
  9.         }).start();  
  10.         threadExecute(business, "main");  
  11.     }     
  12.     public static void threadExecute(Business business, String threadType) {  
  13.         for(int i = 0; i < 100; i++) {  
  14.             try {  
  15.                 if("main".equals(threadType)) {  
  16.                     business.main(i);  
  17.                 } else {  
  18.                     business.sub(i);  
  19.                 }  
  20.             } catch (InterruptedException e) {  
  21.                 e.printStackTrace();  
  22.             }  
  23.         }  
  24.     }  
  25. }  
  26. class Business {  
  27.     private boolean bool = true;  
  28.     public synchronized void main(int loop) throws InterruptedException {  
  29.         while(bool) {  
  30.             this.wait();  
  31.         }  
  32.         for(int i = 0; i < 100; i++) {  
  33.             System.out.println("main thread seq of " + i + ", loop of " + loop);  
  34.         }  
  35.         bool = true;  
  36.         this.notify();  
  37.     }     
  38.     public synchronized void sub(int loop) throws InterruptedException {  
  39.         while(!bool) {  
  40.             this.wait();  
  41.         }  
  42.         for(int i = 0; i < 10; i++) {  
  43.             System.out.println("sub thread seq of " + i + ", loop of " + loop);  
  44.         }  
  45.         bool = false;  
  46.         this.notify();  
  47.     }  
  48. }  

 

        大家注意到没有,在调用wait方法时,都是用while判断条件的,而不是if,在wait方法说明中,也推荐使用while,因为在某些特定的情况下,线程有可能被假唤醒,使用while会循环检测更稳妥。

更多信息请查看 java进阶网 http://www.javady.com

分享到:
评论

相关推荐

    java多线程编程总结

    Java线程:并发协作-生产者消费者模型 Java线程:并发协作-死锁 Java线程:volatile关键字 Java线程:新特征-线程池 Java线程:新特征-有返回值的线程 Java线程:新特征-锁(上) Java线程:新特征-锁(下) Java...

    Java多线程编程总结

    Java线程:并发协作-生产者消费者模型 Java线程:并发协作-死锁 Java线程:volatile关键字 Java线程:新特征-线程池 Java线程:新特征-有返回值的线程 Java线程:新特征-锁(上) Java线程:新特征-锁(下) Java...

    Java 线程总结

    Java线程:并发协作-生产者消费者模型 Java线程:并发协作-死锁 Java线程:volatile关键字 Java线程:新特征-线程池 Java线程:新特征-有返回值的线程 Java线程:新特征-锁(上) Java线程:新特征-锁(下) Java...

    java多线程笔记

    Java线程:概念与原理 2 一、操作系统中线程和进程的概念 2 二、Java中的线程 3 三、Java中关于线程的名词解释...Java线程:并发协作-生产者消费者模型 52 Java线程:并发协作-死锁 55 Java线程:线程之间的数据传递 58

    java实验实验报告 add 实验一:多线程/2.Runnable接口. 实验八:JDBC数据库编程二-存储过程

    重命名 实验三多:线程同步与协作:生产者与消费者 为 实验三:多线程同步与协作:生产者与消费者 实验九:使用jdbc存取大文本 add 实验九:使用jdbc存取大文本. 实验二:多线程打字游戏 add 实验二:多线程...

    【资源免费下载】Java代码积累丨大话设计模式(Java实现版本)、线程协作

    生产者-消费者 设计模式参考《大话设计模式》 工厂简单模式 创造型模式 工厂方法模式 抽象工厂模式 原型模式 建造者模式 单例模式 结构型模式 队列模式 桥接模式 组合模式 装饰模式 外观模式 享元模式 代理模式 行为...

    Java线程指南

    Java线程指南 线程安全与不安全 线程同步synchronized和volatile 线程协作-生产者/消费者模式 Timer和TimerTask 线程池 Callable和Future 锁对象Lock-同步问题更完美的处理方式 Condition-线程通信更高效的方式

    学习Java线程之并发协作生产者消费者模型.pdf

    学习Java线程之并发协作生产者消费者模型.pdf

    操作系统课程设计——模拟生产者与消费者(java)

    在多道程序环境下,进程同步问题十分重要,通过解决“生产者-消费者”问题,可以帮助我们更好的理解进程同步的概念及实现方法。掌握线程创建和终止的方法,加深对线程和进程概念的理解,会用同步与互斥方法实现线程...

    java并发之线程间通信协作.docx

    生产者如果不释放对临界资源的占用权,那么 消费者就无法消费队列中的商品 ,就不会让队列有空间,那么生产者就会一直无限等待下去。因此,一般情况下,当队列满时,会让生产者交出对临界资源的占用权,并进入挂起...

    Java多线程的等待唤醒机制代码演示通俗易懂

    生产者和消费者是一个十分经典的多线程协作模式 **常见方法:** - void wait() 当前线程等待,直到被其他线程唤醒 - void notify() 随机唤醒单个线程 - void notifyAll() 唤醒所有线程

    JAVA作业二生产者消费者问题.docx

    作业二:生产者消费者问题。 要求: (1)生产者与消费者均为独立的线程。 (2)生产者与消费者之间有条不紊的协作进行数据的生产和消费。

    java高手真经 (UML建模+设计模式+面向服务架构) 卷8

    pattern/src/creation/singleton //11.3单例模式 pattern/src/creation/builder //11.4建造者模式 pattern/src/creation/prototype //11.5原型模式 pattern/src/structure/adapter //12.1适配器模式 pattern/src/...

    java高手真经 (UML建模+设计模式+面向服务架构) 卷1

    状态图——Java线程类Thread uml/activity_bug.mdl //06.活动图——Bug管理系统 uml/activity_atm.mdl //06.活动图——ATM机存取款 uml/activity_thread.mdl //06.活动图——Java线程类Thread uml/sequence_...

    java高手真经 (UML建模+设计模式+面向服务架构) 卷9

    pattern/src/creation/singleton //11.3单例模式 pattern/src/creation/builder //11.4建造者模式 pattern/src/creation/prototype //11.5原型模式 pattern/src/structure/adapter //12.1适配器模式 pattern/src/...

    java高手真经 (UML建模+设计模式+面向服务架构) 卷3

    状态图——Java线程类Thread uml/activity_bug.mdl //06.活动图——Bug管理系统 uml/activity_atm.mdl //06.活动图——ATM机存取款 uml/activity_thread.mdl //06.活动图——Java线程类Thread uml/sequence_...

    java高手真经 (UML建模+设计模式+面向服务架构) 卷6

    pattern/src/creation/singleton //11.3单例模式 pattern/src/creation/builder //11.4建造者模式 pattern/src/creation/prototype //11.5原型模式 pattern/src/structure/adapter //12.1适配器模式 pattern/src/...

    个人总结的深入java多线程开发

    1)演示简单的消费者和生产者的例子: 17 2)管道的读写流处理方式 19 3)重要的演示死锁的问题—哲学家就餐问题 20 4)终止多线程程序的两种方式(轮询访问变量和interrupt方法) 23 四Concurrent包详解 25 1)Executor...

Global site tag (gtag.js) - Google Analytics