java实现生产者/消费者的三种方式
package com.wenki.thread;
import java.util.LinkedList;
import java.util.concurrent.LinkedBlockingQueue;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.locks.Condition;
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantLock;
public class ProductAndConsume {
public static void main(String[] args) {
ProductAndConsume o = new ProductAndConsume();
// Storage storage = o.new StorageOne();
// Storage storage = o.new StorageTwo();
Storage storage = o.new StorageThree();
Consumer consumer1 = o.new Consumer(storage);
Consumer consumer2 = o.new Consumer(storage);
Producter producter1 = o.new Producter(storage);
Producter producter2 = o.new Producter(storage);
Producter producter3 = o.new Producter(storage);
consumer1.start();
consumer2.start();
producter1.start();
producter2.start();
producter3.start();
}
class Producter extends Thread{
Storage storage;
public Producter(Storage storage) {
this.storage = storage;
}
public void product(){
this.storage.product();
}
@Override
public void run() {
for(;;){
product();
}
}
}
class Consumer extends Thread{
Storage storage;
public Consumer(Storage storage){
this.storage = storage;
}
public void consume(){
this.storage.consume();
}
@Override
public void run() {
for(;;){
consume();
}
}
}
interface Storage{
int MAX_SIZE = 100;
LinkedList<Object> list = new LinkedList<Object>();
public abstract void product();
public abstract void consume();
}
//第一种方式 wait() + notify()
class StorageOne implements Storage{
@Override
public void product() {
synchronized (list) {
while(list.size() == MAX_SIZE){
try {
list.wait();
} catch (InterruptedException e) {
e.printStackTrace();
}
}
if(list.add(new Object())){
System.out.println("生产 ### 产品数量 : " + list.size());
//通知消费者可以继续消费
list.notifyAll();
}
try {
TimeUnit.MILLISECONDS.sleep(100);
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
}
@Override
public void consume() {
synchronized (list) {
while(list.size() == 0){
try {
list.wait();
} catch (InterruptedException e) {
e.printStackTrace();
}
}
if(list.remove() != null){
System.out.println("消费 ### 产品数量: " + list.size());
//通知生产者可以继续生产
list.notifyAll();
}
try {
TimeUnit.MILLISECONDS.sleep(100);
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
}
}
//第二种方式 await() + signal()
class StorageTwo implements Storage{
Lock lock = new ReentrantLock();
Condition fully = lock.newCondition();
Condition empty = lock.newCondition();
@Override
public void product() {
lock.lock();
try{
while(list.size() == MAX_SIZE){
fully.await();
}
if(list.add(new Object())){
System.out.println("生产 ### 产品数量 : " + list.size());
empty.signalAll();
}
TimeUnit.MILLISECONDS.sleep(100);
} catch (InterruptedException e) {
e.printStackTrace();
}finally {
lock.unlock();
}
}
@Override
public void consume() {
lock.lock();
try{
while(list.size() == 0){
empty.await();
}
if(list.remove() != null){
System.out.println("消费 ### 产品数量: " + list.size());
fully.signalAll();
}
TimeUnit.MILLISECONDS.sleep(100);
} catch (InterruptedException e) {
e.printStackTrace();
}finally{
lock.unlock();
}
}
}
//第三种 BlockingQueue 阻塞队列
class StorageThree implements Storage{
LinkedBlockingQueue<Object> list = new LinkedBlockingQueue<Object>(MAX_SIZE);
@Override
public void product() {
try {
list.put(new Object());
System.out.println("生产 ### 产品数量 : " + list.size());
TimeUnit.MILLISECONDS.sleep(100);
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
@Override
public void consume() {
try {
if(list.take() != null){
System.out.println("消费 ### 产品数量: " + list.size());
}
TimeUnit.MILLISECONDS.sleep(100);
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
}
}
java实现生产者/消费者的三种方式的更多相关文章
- java 实现md5加密的三种方式与解密
java 实现md5加密的三种方式 CreateTime--2018年5月31日15点04分 Author:Marydon 一.解密 说明:截止文章发布,Java没有实现解密,但是已有网站可以免费 ...
- java中遍历集合的三种方式
第一种遍历集合的方式:将集合变为数组 package com.lw.List; import java.util.ArrayList; import java.util.List; import ja ...
- java加载配置文件的三种方式
比如我们要加载db.properties文件 如图: 比如我们要加载source目录下的db.properties文件.就有以下几种方式 第一种是文件io流: public static void l ...
- JAVA实现Base64编码的三种方式
摘要: Javabase64编码的三种方式 有如下三种方式: 方式一:commons-codec.jar Java代码 1. String base64String="whuang12 ...
- Java通过JDBC连接数据库的三种方式!!!并对数据库实现增删改查
前言 java连接数据库完整流程为: 1,获得驱动(driver),数据库连接(url),用户名(username),密码(password)基本信息的三种方式. 2,通过获得的信息完成JDBC实现连 ...
- java实现HTTP请求的三种方式
目前JAVA实现HTTP请求的方法用的最多的有两种:一种是通过HTTPClient这种第三方的开源框架去实现.HTTPClient对HTTP的封装性比较不错,通过它基本上能够满足我们大部分的需求,Ht ...
- 【转载】java实现HTTP请求的三种方式
目前JAVA实现HTTP请求的方法用的最多的有两种:一种是通过HTTPClient这种第三方的开源框架去实现.HTTPClient对HTTP的封装性比较不错,通过它基本上能够满足我们大部分的需求,Ht ...
- Java 实现线程安全的三种方式
一个程序在运行起来的时候会转换成进程,通常含有多个线程. 通常情况下,一个进程中的比较耗时的操作(如长循环.文件上传下载.网络资源获取等),往往会采用多线程来解决. 比如显示生活中,银行取钱问题.火车 ...
- HTTP:Java实现HTTP请求的三种方式
目前JAVA实现HTTP请求的方法用的最多的有两种: 一种是通过HTTPClient这种第三方的开源框架去实现.HTTPClient对HTTP的封装性比较不错,通过它基本上能够满足我们大部分的需求,H ...
随机推荐
- python全栈开发-常用模块的一些应用
一.random模块详解 1.概述 首先我们看到这个单词是随机的意思,他在python中的主要用于一些随机数,或者需要写一些随机数的代码,下面我们就来整理他的一些用法 2.常用方法 1. random ...
- 新概念英语(1-5)Nice to meet you.
Is Chang-woo Chinese? Blake:Good morning. B:Good morning, Mr Blake. Blake:This is Miss Sophie Dupont ...
- 查看eclipse ADT SDK JDK版本号
一.查看eclipsea版本号: 启动eclipse,Help > About Eclipse SDK,在eclipse SDK对话框下面就有Eclipse SDK Version:4.2.0这 ...
- Python学习之再议row_input
再议raw_input birth = raw_input('birth: ') if birth < 2000: print '00前' else: print '00后' 运行结果: bir ...
- 【webGL入门2】点线面的绘制
用js绘制webGL的点: THREE.Vector3 = function ( x, y, z ) { //用THREE声明的变量都是全局变量.this.x = x || 0;this.y = y ...
- HTML-----<a>、<table>、<form>解析
超链接 anchor 锚 <a href="url">内容</a> Href Hypertext reference 引用 URL(Uniform Re ...
- python JavaScript
JavaScript 一. JavaScript Javascript 在开发中绝大多数情况是基于对象的.也是面向对象的. a. JavaScript的引入方式 1 2 3 4 5 6 7 #直接编写 ...
- JavaScript 对图像进行(追加,插入,替换,删除)
JavaScript 对图像进行(追加,插入,替换,删除) 本次所学内容: document.querySelector('.container') 这个是可以查找单个[id标签和class标签] d ...
- Linux安装git和maven的详细过程
一.使用yum安装git 当前安装环境是centos6.5 由于在CentOS6的yum源中已经有git的版本了,可以直接使用yum源进行安装 yum -y install git 由于centos6 ...
- SpringBoot(七):集成DataSource 与 Druid监控配置
绑定DataSource:Spring Boot默认的数据源是:org.apache.tomcat.jdbc.pool.DataSource,Druid是Java语言中最好的数据库连接池,并且能够提供 ...