这是个线程同步的经典例子,源代码如下:

  1. <span style="font-size:16px;">package demo.thread;
  2. /**
  3. *经典生产者与消费者问题:生产者不断的往仓库中存放产品,消费者从仓库中消费产品。
  4. *其中生产者和消费者都可以有若干个。仓库容量有限,库满时不能存放,库空时不能取产品
  5. */
  6. public class ProducersAndConsumers {
  7. public static void main(String[] args) {
  8. Storage storage = new Storage();
  9. Thread consumer = new Thread(new Consumer(storage));
  10. consumer.setName("消费者");
  11. Thread producer = new Thread(new Producer(storage));
  12. producer.setName("生产者");
  13. consumer.start();
  14. producer.start();
  15. }
  16. }
  17. /**
  18. * 消费者
  19. */
  20. class Consumer implements Runnable {
  21. private Storage storage;
  22. public Consumer(Storage storage) {
  23. this.storage = storage;
  24. }
  25. @Override
  26. public void run() {
  27. storage.pop();
  28. }
  29. }
  30. /**
  31. * 生产者
  32. */
  33. class Producer implements Runnable {
  34. private Storage storage;
  35. public Producer(Storage storage) {
  36. this.storage = storage;
  37. }
  38. @Override
  39. public void run() {
  40. Product product = new Product("090505105", "电话");
  41. storage.push(product);
  42. }
  43. }
  44. /**
  45. * 产品类
  46. */
  47. class Product {
  48. private String id;// 产品id
  49. private String name;// 产品名称
  50. public Product(String id, String name) {
  51. this.id = id;
  52. this.name = name;
  53. }
  54. @Override
  55. public String toString() {
  56. return "(产品ID:" + id + " 产品名称:" + name + ")";
  57. }
  58. public String getId() {
  59. return id;
  60. }
  61. public void setId(String id) {
  62. this.id = id;
  63. }
  64. public String getName() {
  65. return name;
  66. }
  67. public void setName(String name) {
  68. this.name = name;
  69. }
  70. }
  71. /**
  72. *仓库
  73. */
  74. class Storage {
  75. // 仓库容量为10
  76. private Product[] products = new Product[10];
  77. private int top = 0;
  78. // 生产者往仓库中放入产品
  79. public synchronized void push(Product product) {
  80. while (top == products.length) {
  81. try {
  82. wait();//仓库已满,等待
  83. } catch (InterruptedException e) {
  84. // TODO Auto-generated catch block
  85. e.printStackTrace();
  86. }
  87. }
  88. //把产品放入仓库
  89. products[top++] = product;
  90. System.out.println(Thread.currentThread().getName() + " 生产了产品"
  91. + product);
  92. notifyAll();//唤醒等待线程
  93. }
  94. // 消费者从仓库中取出产品
  95. public synchronized Product pop() {
  96. while (top == 0) {
  97. try {
  98. wait();//仓库空,等待
  99. } catch (InterruptedException e) {
  100. // TODO Auto-generated catch block
  101. e.printStackTrace();
  102. }
  103. }
  104. //从仓库中取产品
  105. --top;
  106. Product p = new Product(products[top].getId(), products[top].getName());
  107. products[top] = null;
  108. System.out.println(Thread.currentThread().getName() + " 消费了产品" + p);
  109. notifyAll();//唤醒等待线程
  110. return p;
  111. }
  112. }
  113. </span>

运行结果:

生产者 生产了产品(产品ID:090505105 产品名称:电话)
消费者 消费了产品(产品ID:090505105 产品名称:电话)

java多线程总结六:经典生产者消费者问题实现的更多相关文章

  1. Java 多线程学习笔记:生产者消费者问题

    前言:最近在学习Java多线程,看到ImportNew网上有网友翻译的一篇文章<阻塞队列实现生产者消费者模式>.在文中,使用的是Java的concurrent包中的阻塞队列来实现.在看完后 ...

  2. java 多线程并发系列之 生产者消费者模式的两种实现

    在并发编程中使用生产者和消费者模式能够解决绝大多数并发问题.该模式通过平衡生产线程和消费线程的工作能力来提高程序的整体处理数据的速度. 为什么要使用生产者和消费者模式 在线程世界里,生产者就是生产数据 ...

  3. java多线程系列15 设计模式 生产者 - 消费者模式

    生产者-消费者 生产者消费者模式是一个非常经典的多线程模式,比如我们用到的Mq就是其中一种具体实现 在该模式中 通常会有2类线程,消费者线程和生产者线程 生产者提交用户请求 消费者负责处理生产者提交的 ...

  4. Java多线程之并发协作生产者消费者设计模式

    两个线程一个生产者个一个消费者 需求情景 两个线程,一个负责生产,一个负责消费,生产者生产一个,消费者消费一个 涉及问题 同步问题:如何保证同一资源被多个线程并发访问时的完整性.常用的同步方法是采用标 ...

  5. Java多线程—阻塞队列和生产者-消费者模式

    阻塞队列支持生产者-消费者这种设计模式.该模式将“找出需要完成的工作”与“执行工作”这两个过程分离开来,并把工作项放入一个“待完成“列表中以便在随后处理,而不是找出后立即处理.生产者-消费者模式能简化 ...

  6. “全栈2019”Java多线程第六章:中断线程interrupt()方法详解

    难度 初级 学习时间 10分钟 适合人群 零基础 开发语言 Java 开发环境 JDK v11 IntelliJ IDEA v2018.3 文章原文链接 "全栈2019"Java多 ...

  7. Java阻塞队列(BlockingQueue)实现 生产者/消费者 示例

    Java阻塞队列(BlockingQueue)实现 生产者/消费者 示例 本文由 TonySpark 翻译自 Javarevisited.转载请参见文章末尾的要求. Java.util.concurr ...

  8. 【Java并发编程】:生产者—消费者模型

    生产者消费者问题是线程模型中的经典问题:生产者和消费者在同一时间段内共用同一存储空间,生产者向空间里生产数据,而消费者取走数据. 这里实现如下情况的生产--消费模型: 生产者不断交替地生产两组数据“姓 ...

  9. java线程之多个生产者消费者

    温故一下上一节所学习的生产者消费者代码: 两个线程时: 通过标志位flag的if判断和同步函数互斥较好解决两个线程,一个生产者.一个消费者交替执行的功能 类名:ProducterConsumerDem ...

随机推荐

  1. android中setOnClickListener的那点事

    最近在写代码中,发现在xml文件设置了android:clickable="false",之后这个View还是可点的. 后来发现,是代码中对View设置了监听事件(setOnCli ...

  2. windows 一个进程可以允许最大的线程数

    默认情况下,一个线程的栈要预留1M的内存空间 而一个进程中可用的内存空间只有2G,所以理论上一个进程中最多可以开2048个线程 但是内存当然不可能完全拿来作线程的栈,所以实际数目要比这个值要小. 你也 ...

  3. SOD-80 LL34 DL-35 (2.7~75V)贴片稳压二极管【worldsing 笔记

    ¨ Silicon Planar Zener Diodes ¨ In Mini-MELF case especially for automatic insertion. ¨ The Zener vo ...

  4. 在线性级别时间内找出无序序列中的第k个元素

    在一个无序序列中找出第k个元素,对于k很小或者很大时可以采取特殊的方法,比如用堆排序来实现 .但是对于与序列长度N成正比的k来说,就不是一件容易的事了,可能最容易想到的就是先将无序序列排序再遍历即可找 ...

  5. Oracle管道函数示例

    Oracle的管道函数需要定义下面的三样: Record/Object Type:定义一个Record或Object类型的变量,这个变量用于表示返回结果集的一行数据,有点像C#中的DataRow类. ...

  6. MySQL主从复制技术(纯干货)

    1.复制配置     主机一定要开启二进制日志(这里建议配置RBR)     每个主机和每个从机一定要配置一个位移的id,即server-id     每个从机配置一定要包含主机名称,日志名称,和位置 ...

  7. <转>linux 下stm32开发环境安装

    传送门: http://www.eefocus.com/marianna/blog/13-10/298454_7e04f.html http://blog.sina.com.cn/s/blog_643 ...

  8. Mysql 培训

     1. Mysql 培训 1.1. 培训目的 本文档是针对MySQL 数据库方面的基础培训,为了使项目组成员能够达到使用MySQL 数据库的目的. 1.2. 培训对象 开发者 1.3. 经常使用词及符 ...

  9. iOS开发:AVPlayer实现流音频边播边存

    1. AVPlayer简介 AVPlayer存在于AVFoundation中,可以播放视频和音频,可以理解为一个随身听 AVPlayer的关联类: AVAsset:一个抽象类,不能直接使用,代表一个要 ...

  10. android---APN切换

    android手机客户端在上传文件时,有时候会一直失败,其可能的原因是APN的设置.wap下的成功率极低,所以在进行文件上传时最好设置下 apn为net形式.下面是我在网上找的一些代码,是由wap转n ...