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

  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. C++学习笔记(十三):类、包和接口

    看标题感觉这些术语和C++没有什么关系啊,包和接口是Java等语言加入的概念. 这篇文章是基于Java等语言的新概念(相对于C++来说的新概念)来看C++的设计思路. 类: C++中,一个文件可以包含 ...

  2. MSSQLSERVER数据库- 数据类型Timestamp

    Timestamp:亦称时间戳数据类型,它提供数据库范围内的惟一值,反应数据库中数据修改的相对顺序,相当于一个单调上升的计数器.当它所定义的列在更新或者插入数据行时,此列的值会被自动更新,一个计数值将 ...

  3. 【STL源码学习】STL算法学习之四

    排序算法是STL算法中相当常用的一个类别,包括部分排序和全部排序算法,依据效率和应用场景进行选择. 明细: sort 函数原型: template <class RandomAccessIter ...

  4. Android 实现ListView的A-Z字母排序和过滤搜索功能,实现汉字转成拼音

    转载:http://blog.csdn.net/xiaanming/article/details/12684155 转载请注明出处:http://blog.csdn.net/xiaanming/ar ...

  5. 利用Chrome模拟访问移动端网页

    很多网站都通过User-Agent来判断浏览器类型,如果是3G手机,显示手机页面内容,如果是普通浏览器,显示普通网页内容. 谷歌Chrome浏览器,可以很方便地用来当3G手机模拟器.在Windows的 ...

  6. SQLyog MySQL GUI 11.13 Ultimate 中文破解版【转载】

    SQLyog是一个易于使用的.快速而简洁的图形化管理MYSQL数据库的工具,它能够在任何地点有效地管理你的数据库! SQLyog MySQL GUI是我常用的一个桌面工具,功能强大,让你有使用MSSQ ...

  7. node.js在windows下的学习笔记(5)---用NODE.JS创建服务器和客户端

    //引入http模块 var http = require('http'); //调用http的createServer的方法,这个方法有一个回调函数,这个回调数 //的作用是当有请求发送给服务器的时 ...

  8. android EditText输入变化事件详解

    editText.addTextChangedListener(new TextWatcher(){ public void afterTextChanged(Editable s) {    // ...

  9. android学习日记28--Android中常用设计模式总结

    一.综述 设计模式,根据前人经验总结出常见软件工程问题的解决思想套路.GoF一共归纳了23种设计模式,当然还有人扩充,不止这些.设计模式主要利用面向对象语言的特性,而android 的设计主要用JAV ...

  10. mysql 高可用方案漫谈(一) 转阿里云

    https://yq.aliyun.com/articles/2237?spm=5176.blog2238.yqblogcon1.7.e5fyEy#index_section