java生产者-消费者

(一)、问题的引出

有一个数据存储空间,划分为两部分,一部分用于存储人的姓名,另一部分用于存储人的性别;

我们的应用包含两个线程,一个线程不停向数据存储空间添加数据(生产者),另一个线程从数据空间取出数据(消费者);

因为线程的不确定性,存在于以下两种情况:

1.若生产者线程刚向存储空间添加了人的姓名还没添加人的性别,CPU就切换到了消费者线程,消费者线程把姓名和上一个人的性别联系到一起;

2.生产者放了若干数据,消费者才开始取数据,或者是消费者取完一个数据,还没等到生产者放入新的数据,又重复的取出已取过的数据;

/**
 * 模拟数据区域
 *
 */
class Person{

	private String name;

	private String sex;

	public String getName() {
		return name;
	}

	public void setName(String name) {
		this.name = name;
	}

	public String getSex() {
		return sex;
	}

	public void setSex(String sex) {
		this.sex = sex;
	}
}
/**
 * 生产者
 */
class Producer implements Runnable{

	private Person p;

	public Producer(Person p){
		this.p = p;
	}

	public void run() {
		for (int i = 0; i < 100; i++) {
			if(i % 2== 0){
				p.setName("大哥");
				try {
					Thread.sleep(2);
				} catch (InterruptedException e) {
					e.printStackTrace();
				}
				p.setSex("男");
			}else{
				p.setName("小姐");
				try {
					Thread.sleep(5);
				} catch (InterruptedException e) {
					e.printStackTrace();
				}
				p.setSex("女");
			}
		}
	}
}

class Consumer implements  Runnable{
	private Person p;

	public Consumer(Person p){
		this.p = p;
	}
	public void run() {
		for (int i = 0; i < 100; i++) {
			String name = p.getName();
			String sex = p.getSex();
			System.out.println(name+" --> " + sex);
		}
	}

}
public class Producer_ConsumerDemo {
	public static void main(String[] args) {
		Person p = new Person();

		new Thread(new Producer(p)).start();
		new Thread(new Consumer(p)).start();
	}
}

上机运行上述程序,可见问题的结果。每次运行的结果都不一样,但是都是有问题的结果。

接下来的代码是对上述程序问题的解决。当然了,就是解决:  1.若生产者线程刚向存储空间添加了人的姓名还没添加人的性别,CPU就切换到了消费者线程,消费者线程把姓名和上一个人的性别联系到一起;2.生产者放了若干数据,消费者才开始取数据,或者是消费者取完一个数据,还没等到生产者放入新的数据,又重复的取出已取过的数据;

只不过,问题1反映在本例的男女性别的混乱。问题2反映在本例的“大哥----男”和“小姐----女”不是按照我们程序设计的交替出现(生产消费)。

看改进代码:

/**
 * 模拟数据存储区域
 *
 */
/*
 *  设置名字和性别同步完成,
 */
class Person {

	private String name;

	private String sex;

	// 表示存储区域是否为空。
	//存储区为空,消费者停止消费,生产者工作;存储区不为空,生产者停止生产,消费者开始消费
	private Boolean Empty = Boolean.TRUE;

/**
* 生产者生产人(姓名,性别)
*
*/
	public void set(String name, String sex) {
		synchronized (this) {
			// if(Empty)
			while (!Empty.equals(Boolean.TRUE)) {
				// 区域不为空
				// 此时生产者应该停下来,等着消费者“消费”
				try {
					this.wait();// 等待 消费者消费
				} catch (InterruptedException e) {
					e.printStackTrace();
				}

			}
			this.name = name;
			//
			this.sex = sex;
			// 生产者生产之后,应该修改存储区域的状态以显示其存储空间有产品可供“消费”
			Empty = Boolean.FALSE;// 不为空
			this.notify();// 唤醒消费者起来“消费”,可以理解为读存储空间里的数据
		}

	}

/**
 * 消费者消费(姓名,性别)
 */
	public void get() {
		synchronized (this) {
			// 存储区域为空
			while (!Empty.equals(Boolean.FALSE)) {
				try {
					this.wait();// 等着生产者去生产
				} catch (InterruptedException e) {
					e.printStackTrace();
				}

			}
			String name = getName();
			String sex = getSex();
			System.out.println(name + " --> " + sex);

			// 消费完成,应该修改存储区域的状态

			Empty = Boolean.TRUE;// 空了
			this.notify();// 唤醒生产者
		}

	}

	public String getName() {
		return name;
	}

	public void setName(String name) {
		this.name = name;
	}

	public String getSex() {
		return sex;
	}

	public void setSex(String sex) {
		this.sex = sex;
	}
}

/**
 * 生产者
 */
class Producer implements Runnable {

	private Person p;

	public Producer(Person p) {
		this.p = p;
	}

	public void run() {
		for (int i = 0; i < 100; i++) {
			if (i % 2 == 0) {
				p.set("大哥", "男");
			} else {
				p.set("小姐", "女");
			}
		}
	}
}

class Consumer implements Runnable {
	private Person p;

	public Consumer(Person p) {
		this.p = p;
	}

	public void run() {
		for (int i = 0; i < 100; i++) {
			p.get();
		}
	}

}

public class Producer_ConsumerDemo {
	public static void main(String[] args) {
		Person p = new Person();
        //加下面的话就相当于两个生产者两个消费者
		//new Thread(new Producer(p)).start();
		//new Thread(new Consumer(p)).start();

		new Thread(new Producer(p)).start();
		new Thread(new Consumer(p)).start();
	}
}
</pre><pre name="code" class="java"><span style="white-space:pre"></span>

程序结果:

大哥 --> 男

小姐 --> 女

大哥 --> 男

小姐 --> 女

..................

注意:

wait():让当前线程放弃监视器进入等待,直到其他线程调用同一个监视器并调用notify()或notifyAll()为止。

notify():唤醒在同一对象监听器中调用wait方法的第一个线程。

notifyAll():唤醒在同一对象监听器中调用wait方法的所有线程。

wait()、notify()、notifyAll(),这三个方法属于Object 不属于 Thread,这三个方法必须由同步监视对象来调用,两种情况:

1.synchronized修饰的方法,因为该类的默认实例(this)就是同步监视器,所以可以在同步方法中调用这三个方法;

2.synchronized修饰的同步代码块,同步监视器是括号里的对象,所以必须使用该对象(同步监视器)调用这三个方法;

但是下面问题又来了:要是我们使用的是Lock对象来保证同步的,系统中不存在隐式的同步监视器对象,那么就不能使用者三个方法了,那该咋办呢?

 解决办法:

Lock代替了同步方法或同步代码块,就用Condition来代替同步监视器的功能;

Condition对象通过Lock对象的newCondition()方法创建;

里面方法包括:

await():  等价于同步监听器的wait()方法;

signal(): 等价于同步监听器的notify()方法;

signalAll(): 等价于同步监听器的notifyAll()方法;

具体实现情况看下面的代码:

import java.util.concurrent.locks.Condition;
import java.util.concurrent.locks.ReentrantLock;

/**
 * 模拟数据区域
 *
 * 可重入锁没有同步监听对象,咋办呢?
 *
 * Lock 替代了 synchronized 方法和语句的使用,Condition 替代了 Object 监视器方法的使用。
 */
class Person {

	// 创建可重入锁对象
	private final ReentrantLock lock = new ReentrantLock();

	private  final  Condition con = lock.newCondition();
	private String name;

	private String sex;

	// 表示存储区域是否为空
	private Boolean Empty = Boolean.TRUE;

	/**
	 * 生产
	 *
	 */
	public void set(String name, String sex) {
		lock.lock();
		while(!Empty.equals(Boolean.TRUE)){
			//表示不空状态
			try {
				con.await();
			} catch (InterruptedException e) {
				e.printStackTrace();
			}
		}
		try {
			this.name = name;
			Thread.sleep(1);
			this.sex = sex;
			isEmpty = Boolean.FALSE;
			con.signal();
		} catch (InterruptedException e) {
			e.printStackTrace();
		} finally {
			lock.unlock();
		}

	}

	/**
	 * 消费
	 */
	public void get() {
		lock.lock();
		while(!Empty.equals(Boolean.FALSE)){
			//存储区域为空,消费者应该等着
			try {
				con.await();
			} catch (InterruptedException e) {
				e.printStackTrace();
			}
		}
		try {
			String name = getName();
			String sex = getSex();
			System.out.println(name + " --> " + sex);
			//
			Empty = Boolean.TRUE;
			con.signal();
		} finally {
			lock.unlock();
		}

	}

	public String getName() {
		return name;
	}

	public void setName(String name) {
		this.name = name;
	}

	public String getSex() {
		return sex;
	}

	public void setSex(String sex) {
		this.sex = sex;
	}
}

/**
 * 生产者
 */
class Producer implements Runnable {

	private Person p;

	public Producer(Person p) {
		this.p = p;
	}

	public void run() {
		for (int i = 0; i < 100; i++) {
			if (i % 2 == 0) {
				p.set("大哥", "男");
			} else {
				p.set("小姐", "女");
			}
		}
	}
}

class Consumer implements Runnable {
	private Person p;

	public Consumer(Person p) {
		this.p = p;
	}

	public void run() {
		for (int i = 0; i < 100; i++) {
			p.get();
		}
	}

}

public class Producer_ConsumerDemo {
	public static void main(String[] args) {
		Person p = new Person();

		new Thread(new Producer(p)).start();
		new Thread(new Consumer(p)).start();

	}
}

程序结果:

大哥 --> 男

小姐 --> 女

大哥 --> 男

小姐 --> 女

..................

时间: 2024-08-04 03:56:26

java生产者-消费者的相关文章

java生产者消费者模型

import java.util.Queue;import java.util.concurrent.LinkedBlockingQueue; public class Consumer extends Thread {    private String product;    private Queue<String> storeHouse = new LinkedBlockingQueue<String>();        public Consumer(){      

基于Java 生产者消费者模式(详细分析)

本文目录:1.等待.唤醒机制的原理2.Lock和Condition3.单生产者单消费者模式4.使用Lock和Condition实现单生产单消费模式5.多生产多消费模式(单面包)6.多生产多消费模式 生产者消费者模式是多线程中最为常见的模式:生产者线程(一个或多个)生成面包放进篮子里(集合或数组),同时,消费者线程(一个或多个)从篮子里(集合或数组)取出面包消耗.虽然它们任务不同,但处理的资源是相同的,这体现的是一种线程间通信方式. 本文将先说明单生产者单消费者的情况,之后再说明多生产者多消费者模

java 生产者消费者问题

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

java 生产者消费者问题 并发问题的解决(转)

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

超详细的Java 生产者消费者模式分析

生产者消费者模式是多线程中最为常见的模式:生产者线程(一个或多个)生成面包放进篮子里(集合或数组),同时,消费者线程(一个或多个)从篮子里(集合或数组)取出面包消耗.虽然它们任务不同,但处理的资源是相同的,这体现的是一种线程间通信方式. 本文将先说明单生产者单消费者的情况,之后再说明多生产者多消费者模式的情况.还会分别使用wait()/nofity()/nofityAll()机制.lock()/unlock()机制实现这两种模式. 在开始介绍模式之前,先解释下wait().notify()和no

JAVA生产者消费者模式的实现

春节回了趟老家,又体验了一次流水席,由于桌席多,导致上菜慢,于是在等待间,总结了一下出菜流程的几个特点: 1.有多个灶台,多个灶台都在同时做菜出来. 2.做出来的菜,会有专人用一个托盘端出来,每次端出来的菜(是同一个菜品)的数量不等. 3.由于端出来的菜可能不能满足所有的桌数,所以,端菜人可能会随机选择几桌(一般是就近原则,或者是主桌先端过去)上菜,其余的桌数继续等待后面的端菜人出来. 以上3个条件,完全就是一个生产者消费者模式的场景,于是,把生产者消费者模式先来实现一下,然后再分析如何才能更快

java生产者消费者并发协作

随着职务转变,代码荒废很久了,很多时间都是在沟通需求,作为一名技术员,不写代码就感觉是在自废武功,慢慢颓废了很多,今天重新回顾了下JAVA线程知识,基础知识就不梳理了,网上也很多,主要关键几个状态位(新建.可运行.正在运行.阻塞等)和几个关键方法(sleep.yield.wait.notify.notifyAll等)搞清楚,基础应用知识应该了解的就差不多了,想深入了解的推荐看<JAVA并发编程实战>.废话不多说了,直接上例子,看了下网上的大致思路,结合实际应用时的场景,自己编写了一个例子,不到

java生产者/消费者模式实现——一生产者一消费者(操作值)

胶多不粘话多不甜,直接上代码: 生产者类: /** * Created by 51304 on 2016/2/28. */ public class P { private String lock; public P(String lock){ this.lock = lock; } public void setValue(){ try{ synchronized (this.lock){ if(!ValueObject.value.equals("")){ lock.wait();

java生产者消费者实例

import java.util.Queue;import java.util.LinkedList;public class ProducerConsumerDemo { public static void main(String[] args) { MyQueue q = new MyQueue(); Producer p = new Producer(q); Consumer c = new Consumer(q); p.start(); c.start(); }}class Produ