java 生产者消费者问题_第1页
java 生产者消费者问题_第2页
java 生产者消费者问题_第3页
java 生产者消费者问题_第4页
java 生产者消费者问题_第5页
已阅读5页,还剩9页未读 继续免费阅读

下载本文档

版权说明:本文档由用户提供并上传,收益归属内容提供方,若内容存在侵权,请进行举报或认领

文档简介

1、java 生产者消费者问题引言生产者和消费者问题是线程模型中的经典问题:生产者和消费者在同一时间段内共用同一个存储空间,如下图所示,生产者向空间里存放数据,而消费者取用数据,如果不加以协调可能会出现以下情况:生产者消费者图存储空间已满,而生产者占用着它,消费者等着生产者让出空间从而去除产品,生产者等着消费者消费产品,从而向空间中添加产品。互相等待,从而发生死锁。生产者消费者问题是研究多线程程序时绕不开的经典问题之一,它描述是有一块缓冲区作为仓库,生产者可以将产品放入仓库,消费者则可以从仓库中取走产品。解决生产者/消费者问题的方法可分为两类:(1)采用某种机制保护生产者和消费者之间的同步;(2)

2、在生产者和消费者之间建立一个管道。第一种方式有较高的效率,并且易于实现,代码的可控制性较好,属于常用的模式。第二种管道缓冲区不易控制,被传输数据对象不易于封装等,实用性不强。因此本文只介绍同步机制实现的生产者/消费者问题。同步问题核心在于:如何保证同一资源被多个线程并发访问时的完整性。常用的同步方法是采用信号或加锁机制, 保证资源在任意时刻至多被一个线程访问。Java语言在多线程编程上实现了完全对象化,提供了对同步机制的良好支持。在Java中一共有四种方法支持同 步,其中前三个是同步方法,一个是管道方法。(1)wait() / notify()方法(2)await() / signal()方法

3、(3)BlockingQueue阻塞队列方法(4)PipedInputStream / PipedOutputStream本文只介绍最常用的前两种种,第三、四种暂不做讨论,有兴趣的读者可以自己去网上找答案。一、wait() / notify()方法wait() / nofity()方法是基类Object的两个方法,也就意味着所有Java类都会拥有这两个方法,这样,我们就可以为任何对象实现同步机制。wait()方法:当缓冲区已满/空时,生产者/消费者线程停止自己的执行,放弃锁,使自己处于等等状态,让其他线程执行。notify()方法:当生产者/消费者向缓冲区放入/取出一个产品时,向其他等待的线程

4、发出可执行的通知,同时放弃锁,使自己处于等待状态。代码实现:1、仓库类1. import java.util.LinkedList;   2.  3. /*  4.  * 仓库类Storage实现缓冲区  5.  *   6.  * author zcr  7.  */   8. public class Storage 

5、60; 9.    10.     / 仓库最大存储量   11.     private final int MAX_SIZE = 100;   12.    13.     / 仓库存储的载体   14.    &#

6、160;private LinkedList<Object> list = new LinkedList<Object>();   15.    16.     /* 17.      * 生产num个产品 18.      * param num 生产产品的数量&

7、#160;19.      */ 20.     public void produce(int num)   21.        22.         / 同步代码段   23.      &#

8、160;  synchronized (list)   24.            25.             / 如果仓库剩余容量不足   26.          

9、60;  while (list.size() + num > MAX_SIZE)   27.                28.                 System.ou

10、t.println("【要生产的产品数量】:" + num + " t 【库存量】:"   29.                         + list.size() + "t

11、60;暂时不能执行生产任务!");   30.                 try   31.                    32.    

12、;                 / 由于条件不满足,生产阻塞   33.                     list.wait();   34. &#

13、160;                  35.                 catch (InterruptedException e)   36.     &#

14、160;              37.                     e.printStackTrace();   38.        

15、0;           39.                40.    41.             / 生产条件满足情况下,生产num个产品  

16、60;42.             for (int i = 1; i <= num; +i)   43.                44.      

17、60;          list.add(new Object();   45.                46.    47.             

18、System.out.println("【已经生产产品数】:" + num + "t 【现仓储量为】:" + list.size();   48.    49.             list.notifyAll();   50.     

19、;       51.        52.    53.     /* 54.      * 消费num个产品 55.      * param num 消费产品数量 56.    

20、60; */ 57.     public void consume(int num)   58.        59.         / 同步代码段   60.         synchronized&#

21、160;(list)   61.            62.             / 如果仓库存储量不足   63.             while (list

22、.size() < num)   64.                65.                 System.out.println("【要消费的产品数量】:" + num

23、60;+ " t【库存量】:"   66.                         + list.size() + " t 暂时不能执行生产任务!");   67.  &#

24、160;              try   68.                    69.            

25、60;        / 由于条件不满足,消费阻塞   70.                     list.wait();   71.          

26、          72.                 catch (InterruptedException e)   73.              

27、      74.                     e.printStackTrace();   75.                 &#

28、160;  76.                77.    78.             / 消费条件满足情况下,消费num个产品   79.        

29、60;    for (int i = 1; i <= num; +i)   80.                81.               &

30、#160; list.remove();   82.                83.    84.             System.out.println("【已经消费产品数】:" + num +&

31、#160;" t 【现仓储量为】:" + list.size();   85.    86.             list.notifyAll();   87.            88.   

32、     89.    90.     / get/set方法   91.     public LinkedList<Object> getList()   92.        93.       

33、60; return list;   94.        95.    96.     public void setList(LinkedList<Object> list)   97.        98.     

34、60;   this.list = list;   99.        100.    101.     public int getMAX_SIZE()   102.        103.      &#

35、160;  return MAX_SIZE;   104.        105.  2、生产者1. /*  2.  * 生产者类Producer继承线程类Thread  3.  *   4.  *   5.  * author zcr 6.  *  

36、; 7.  */   8. public class Producer extends Thread   9.    10.     / 每次生产的产品数量   11.     private int num;   12.    13.  

37、   / 所在放置的仓库   14.     private Storage storage;   15.    16.     / 构造函数,设置仓库   17.     public Producer(Storage storage)  

38、0;18.        19.         this.storage = storage;   20.        21.    22.     / 线程run函数   23.    

39、; public void run()   24.        25.         produce(num);   26.        27.    28.     / 调用仓库Storage的生产函数

40、   29.     public void produce(int num)   30.        31.         duce(num);   32.        33.  

41、  34.     / get/set方法   35.     public int getNum()   36.        37.         return num;   38.   

42、60;    39.    40.     public void setNum(int num)   41.        42.         this.num = num;   43.    

43、    44.    45.     public Storage getStorage()   46.        47.         return storage;   48.      &

44、#160; 49.    50.     public void setStorage(Storage storage)   51.        52.         this.storage = storage;   53.   &#

45、160;    54.  3、消费者1. /*  2.  * 消费者类Consumer继承线程类Thread  3.  *   4.  *   5.  * author zcr 6.  *   7.  */   8. public class Consumer 

46、;extends Thread   9.    10.     / 每次消费的产品数量   11.     private int num;   12.    13.     / 所在放置的仓库   14.    

47、60;private Storage storage;   15.    16.     / 构造函数,设置仓库   17.     public Consumer(Storage storage)   18.        19.     

48、;    this.storage = storage;   20.        21.    22.     / 线程run函数   23.     public void run()   24.    &

49、#160;   25.         consume(num);   26.        27.    28.     / 调用仓库Storage的生产函数   29.     public void consume

50、(int num)   30.        31.         storage.consume(num);   32.        33.    34.     / get/set方法   3

51、5.     public int getNum()   36.        37.         return num;   38.        39.    40.    

52、0;public void setNum(int num)   41.        42.         this.num = num;   43.        44.    45.     p

53、ublic Storage getStorage()   46.        47.         return storage;   48.        49.    50.     public vo

54、id setStorage(Storage storage)   51.        52.         this.storage = storage;   53.        54.  4、测试类1. /*  2.  * 

55、;测试类Test  3.  * author zcr 4.  *   5.  */   6. public class Test   7.    8.     public static void main(String args)   9.   &

56、#160;    10.         / 仓库对象   11.         Storage storage = new Storage();   12.    13.        

57、 / 生产者对象   14.         Producer p1 = new Producer(storage);   15.         Producer p2 = new Producer(storage);   16.  &#

58、160;      Producer p3 = new Producer(storage);   17.         Producer p4 = new Producer(storage);   18.         Producer

59、 p5 = new Producer(storage);   19.         Producer p6 = new Producer(storage);   20.         Producer p7 = new Producer(storage);

60、   21.    22.         / 消费者对象   23.         Consumer c1 = new Consumer(storage);   24.         

61、Consumer c2 = new Consumer(storage);   25.         Consumer c3 = new Consumer(storage);   26.    27.         / 设置生产者产品生产数量 &#

62、160; 28.         p1.setNum(10);   29.         p2.setNum(10);   30.         p3.setNum(10);   31.      

63、0;  p4.setNum(10);   32.         p5.setNum(10);   33.         p6.setNum(10);   34.         p7.setNum(80);   35

64、.    36.         / 设置消费者产品消费数量   37.         c1.setNum(50);   38.         c2.setNum(20);   39.    &#

65、160;    c3.setNum(30);   40.    41.         / 线程开始执行   42.         c1.start();   43.         c2

66、.start();   44.         c3.start();   45.         p1.start();   46.         p2.start();   47.     &#

67、160;   p3.start();   48.         p4.start();   49.         p5.start();   50.         p6.start();   51. &#

68、160;       p7.start();   52.        53.  5、结果:1. 【要消费的产品数量】:50     【库存量】:0      暂时不能执行生产任务! 2. 【要消费的产品数量】:20     【库存量】:0 

69、;     暂时不能执行生产任务! 3. 【已经生产产品数】:10     【现仓储量为】:10 4. 【要消费的产品数量】:20     【库存量】:10      暂时不能执行生产任务! 5. 【要消费的产品数量】:50     【库存量】:10     

70、 暂时不能执行生产任务! 6. 【已经生产产品数】:10     【现仓储量为】:20 7. 【已经生产产品数】:10     【现仓储量为】:30 8. 【要消费的产品数量】:50     【库存量】:30      暂时不能执行生产任务! 9. 【已经消费产品数】:20     

71、60;【现仓储量为】:10 10. 【已经生产产品数】:10     【现仓储量为】:20 11. 【已经生产产品数】:10     【现仓储量为】:30 12. 【要消费的产品数量】:50     【库存量】:30      暂时不能执行生产任务! 13. 【已经生产产品数】:10     【现仓

72、储量为】:40 14. 【已经消费产品数】:30      【现仓储量为】:10 15. 【要消费的产品数量】:50     【库存量】:10      暂时不能执行生产任务! 16. 【已经生产产品数】:80     【现仓储量为】:90 17. 【已经消费产品数】:50     

73、60;【现仓储量为】:40 看完上述代码,对wait() / notify()方法实现的同步有了了解。你可能会对Storage类中为什么要定义public void produce(int num);和public void consume(int num);方法感到不解,为什么不直接在生产者类Producer和消费者类Consumer中实现这两个方法,却要调用Storage类中的实现呢?淡定,后文会有解释。我们先往下走。二、await() / signal()方法在JDK5.0之后,Java提供了更加健壮的线程处理机制,包括同步、锁定、线程池等,它们可以实现更细粒度的线程控制。aw

74、ait()和 signal()就是其中用来做同步的两种方法,它们的功能基本上和wait() / nofity()相同,完全可以取代它们,但是它们和新引入的锁定机制Lock直接挂钩,具有更大的灵活性。通过在Lock对象上调用newCondition()方法,将条件变量和一个锁对象进行绑定,进而控制并发程序访问竞争资源的安全。下面来看代码:只需要更新仓库类Storage的代码即可,生产者Producer、消费者Consumer、测试类Test的代码均不需要进行任何更改。仓库类1. import java.util.LinkedList; 2. import java.

75、util.concurrent.locks.Condition; 3. import java.util.concurrent.locks.Lock; 4. import java.util.concurrent.locks.ReentrantLock; 5.  6. /* 7.  * 仓库类Storage实现缓冲区 8.  *  9.  *  10.  * author MONKEY.D.MENG

76、0;2011-03-15 11.  *  12.  */ 13. public class Storage 14.  15.     / 仓库最大存储量 16.     private final int MAX_SIZE = 100; 17.  18.     / 仓库存储的载

77、体 19.     private LinkedList<Object> list = new LinkedList<Object>(); 20.  21.     / 锁 22.     private final Lock lock = new ReentrantLock(); 2

78、3.  24.     / 仓库满的条件变量 25.     private final Condition full = lock.newCondition(); 26.  27.     / 仓库空的条件变量 28.     private final Condition empty&

79、#160;= lock.newCondition(); 29.  30.     / 生产num个产品 31.     public void produce(int num) 32.      33.         / 获得锁 34.    &#

80、160;    lock.lock(); 35.  36.         / 如果仓库剩余容量不足 37.         while (list.size() + num > MAX_SIZE) 38.        &

81、#160; 39.             System.out.println("【要生产的产品数量】:" + num + "/t【库存量】:" + list.size() 40.               

82、0;     + "/t暂时不能执行生产任务!"); 41.             try 42.              43.          

83、60;      / 由于条件不满足,生产阻塞 44.                 full.await(); 45.              46.      &

84、#160;      catch (InterruptedException e) 47.              48.                 e.printStackTrace(); 49. &

85、#160;            50.          51.  52.         / 生产条件满足情况下,生产num个产品 53.         for (int i

86、 = 1; i <= num; +i) 54.          55.             list.add(new Object(); 56.          57.  58. 

87、60;       System.out.println("【已经生产产品数】:" + num + "/t【现仓储量为】:" + list.size(); 59.  60.         / 唤醒其他所有线程 61.        

88、60;full.signalAll(); 62.         empty.signalAll(); 63.  64.         / 释放锁 65.         lock.unlock(); 66.      67.  68.

89、    / 消费num个产品 69.     public void consume(int num) 70.      71.         / 获得锁 72.         lock.lock(); 73. &#

90、160;74.         / 如果仓库存储量不足 75.         while (list.size() < num) 76.          77.           

91、;  System.out.println("【要消费的产品数量】:" + num + "/t【库存量】:" + list.size() 78.                     + "/t暂时不能执行生产任务!"); 79

92、.             try 80.              81.                 / 由于条件不满足,消费阻塞 82.  

93、               empty.await(); 83.              84.             catch (InterruptedException e) 85.   

温馨提示

  • 1. 本站所有资源如无特殊说明,都需要本地电脑安装OFFICE2007和PDF阅读器。图纸软件为CAD,CAXA,PROE,UG,SolidWorks等.压缩文件请下载最新的WinRAR软件解压。
  • 2. 本站的文档不包含任何第三方提供的附件图纸等,如果需要附件,请联系上传者。文件的所有权益归上传用户所有。
  • 3. 本站RAR压缩包中若带图纸,网页内容里面会有图纸预览,若没有图纸预览就没有图纸。
  • 4. 未经权益所有人同意不得将文件中的内容挪作商业或盈利用途。
  • 5. 人人文库网仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对用户上传分享的文档内容本身不做任何修改或编辑,并不能对任何下载内容负责。
  • 6. 下载文件中如有侵权或不适当内容,请与我们联系,我们立即纠正。
  • 7. 本站不保证下载资源的准确性、安全性和完整性, 同时也不承担用户因使用这些下载资源对自己和他人造成任何形式的伤害或损失。

评论

0/150

提交评论