消费者和生产者模式

用继承Thread方式,用wait和notifyAll方法实现。

消费者和生产者模式的特点

1. 什么时候生产:仓库没有满的时候,生产者这可以生产,消费者也可以消费,仓库满的时候停止生产
2. 什么时候消费: 仓库有货的时候消费,没有货不能消费
3. 通知生产:消费者发现没有货,消费者通知生产者生产
4. 通知消费:生产者生产出产品后,通知消费者消费

代码实现

package com.java.javabase.thread.base;

import lombok.AllArgsConstructor;
import lombok.Data;
import lombok.RequiredArgsConstructor;
import lombok.extern.slf4j.Slf4j; /*
1. 什么时候生产:仓库没有满的时候,生产者这可以生产,消费者也可以消费,仓库满的时候停止生产
2. 什么时候消费: 仓库有货的时候消费,没有货不能消费
3. 通知生产:消费者发现没有货,消费者通知生产者生产
4. 通知消费:生产者生产出产品后,通知消费者消费
*/
@Data
@Slf4j
@AllArgsConstructor
@RequiredArgsConstructor
class Dept {
private int capacity;
private int size; /*
@param val 是生产的产品数量
*/
public synchronized void produce(int val) {
//
int remainder = val;
while (remainder > 0) {
if (size >= capacity) {
try {
//仓库满了,等待不在执行生产
wait();
} catch (InterruptedException e) {
e.printStackTrace();
}
}
int incre = (size + remainder) > capacity ? capacity - size : remainder;
size += incre;
remainder -= incre;
log.info("produce {} .depet increment {}, remainder {} ,current size is {}",
val,incre, remainder, size);
//结束生产,通知消费或者生产
notifyAll();
} } public synchronized void custom(int val) {
int remainder = val;
while(remainder>0){
if(size<=0){
try {
//仓库无货,等待不在消费
wait();
} catch (InterruptedException e) {
e.printStackTrace();
}
}
int descre =(size<remainder)?size:remainder;
size -=descre;
remainder -=descre;
log.info("custom {}. depet descrement {}, remainder {} ,current size is {}",
val,descre, remainder, size);
//结束消费,通知消费或者生产
notifyAll();
} }
} @AllArgsConstructor
class Producer {
private Dept dept; public void produce(final int val) {
new Thread() {
@Override
public void run() {
dept.produce(val);
}
}.start();
} } @AllArgsConstructor
class Custom {
private Dept dept; public void custom(final int val) {
new Thread() {
@Override
public void run() {
dept.custom(val);
}
}.start();
} } public class DemoProCustomTest {
public static void main(String[] args) {
Dept dept =new Dept(100,0);
Producer producer =new Producer(dept);
Custom custom =new Custom(dept);
producer.produce(100);
producer.produce(190);
custom.custom(150);
custom.custom(80); }
}

日志结果输出

因为多线程,所以下面的结果只是其中之一的可能性

2019-07-31 19:10:13,297   [Thread-0] INFO  Dept  - produce 100 .depet increment 100, remainder 0 ,current size is 100
2019-07-31 19:10:13,299 [Thread-3] INFO Dept - custom 80. depet descrement 80, remainder 0 ,current size is 20
2019-07-31 19:10:13,299 [Thread-2] INFO Dept - custom 150. depet descrement 20, remainder 130 ,current size is 0
2019-07-31 19:10:13,300 [Thread-1] INFO Dept - produce 190 .depet increment 100, remainder 90 ,current size is 100
2019-07-31 19:10:13,300 [Thread-2] INFO Dept - custom 150. depet descrement 100, remainder 30 ,current size is 0
2019-07-31 19:10:13,300 [Thread-1] INFO Dept - produce 190 .depet increment 90, remainder 0 ,current size is 90
2019-07-31 19:10:13,300 [Thread-2] INFO Dept - custom 150. depet descrement 30, remainder 0 ,current size is 60

java并发:初探消费者和生产者模式的更多相关文章

  1. java多线程-消费者和生产者模式

    /* * 多线程-消费者和生产者模式 * 在实现消费者生产者模式的时候必须要具备两个前提,一是,必须访问的是一个共享资源,二是必须要有线程锁,且锁的是同一个对象 * */ /*资源类中定义了name( ...

  2. springcloud 实现简单的 消费者和生产者 模式(Restfule 的风格)

    一.springcloud 实现简单的 消费者和生产者 模式(Restfule 的风格) 1.实现简单的消费者和生产者 springcloud使用的http协议进行传输数据,也就是说springclo ...

  3. java并发初探ConcurrentSkipListMap

    java并发初探ConcurrentSkipListMap ConcurrentSkipListMap以调表这种数据结构以空间换时间获得效率,通过volatile和CAS操作保证线程安全,而且它保证了 ...

  4. java并发初探ConcurrentHashMap

    java并发初探ConcurrentHashMap Doug Lea在java并发上创造了不可磨灭的功劳,ConcurrentHashMap体现这位大师的非凡能力. 1.8中ConcurrentHas ...

  5. java并发初探ThreadPoolExecutor拒绝策略

    java并发初探ThreadPoolExecutor拒绝策略 ThreadPoolExecuter构造器 corePoolSize是核心线程池,就是常驻线程池数量: maximumPoolSize是最 ...

  6. java并发初探CyclicBarrier

    java并发初探CyclicBarrier CyclicBarrier的作用 CyclicBarrier,"循环屏障"的作用就是一系列的线程等待直至达到屏障的"瓶颈点&q ...

  7. java并发初探CountDownLatch

    java并发初探CountDownLatch CountDownLatch是同步工具类能够允许一个或者多个线程等待直到其他线程完成操作. 当前前程A调用CountDownLatch的await方法进入 ...

  8. java并发初探ReentrantWriteReadLock

    java并发初探ReentrantWriteReadLock ReenWriteReadLock类的优秀博客 ReentrantReadWriteLock读写锁详解 Java多线程系列--" ...

  9. java 模拟实现消费者和生产者问题

    题目要求 用java代码模拟实现:一个人不断往箱子里放苹果,另一个人不断从箱子里取苹果,箱子只能放5个苹果,苹果数量无限.要求不使用java.util.concurrent包中的类. 思路 这道题主要 ...

随机推荐

  1. python练习:使用二分法查找求近似平方根,使用二分法查找求近似立方根。

    python练习:使用二分法查找求近似平方根,使用二分法查找求近似立方根. 重难点:原理为一个数的平方根一定在,0到这个数之间,那么就对这之间的数,进行二分遍历.精确度的使用.通过最高值和最低值确定二 ...

  2. java.lang.ClassNotFoundException: javax.xml.bind.DatatypeConverter 可能是我们运行的java版本过高导致

    最近给自己写了个 SpringBoot 应用程序 顺便练练手,准备把程序过到 树莓派 上,结果登陆报错

  3. python网络爬虫之解析网页的XPath(爬取Path职位信息)[三]

    目录 前言 XPath的使用方法 XPath爬取数据 后言 @(目录) 前言 本章同样是解析网页,不过使用的解析技术为XPath. 相对于之前的BeautifulSoup,我感觉还行,也是一个比较常用 ...

  4. list随机生成数值

    List<int> numbers = Enumerable.Range(5, 10).ToList();

  5. 常用es5和es6语法区别,以及三个点的用法

    链接:https://www.jianshu.com/p/b4d48e9846e7 //三个点 链接:https://blog.csdn.net/qiladuo1207/article/details ...

  6. cf854B Maxim Buys an Apartment

    Maxim Buys an Apartment #include <iostream> #define int long long using namespace std; int n,k ...

  7. 给服务添加路由--ingress

    kind: NamespaceapiVersion: v1metadata:  name: demo-webshell  generateName: demo-webshell  labels:    ...

  8. Nexus-配置vPC 实验一

    配置vPC的步骤:1.配置vPC domain2.配置vPC之间的keepalive link3.配置vPC之间的peer link4.配置vPCs5.确认双方配置一致 拓扑及描述:DC1-N7K-5 ...

  9. ElasticSearch学习记录 - 命令示例

    GET /searchfilmcomments/searchfilmcomments/_search { "query": { "match_all": {} ...

  10. java 类型转换一些相关问题

    猜测:第二句 第四句会出错 结果是第二句和第四句会出错.说明了父类可以向子类类型转换,而不同的子类直接不能类型转换.