标签:多线程
1.多线程-线程间通信-多生产者多消费者问题
多生产者和多消费者。等待唤醒机制。
产生了两个问题:
1.出现了多次连续生产,未消费,或者一个商品被消费多次。
解决:必须要--------每一个被唤醒的线程判断一次标记,所以将if判断改为while判断。
2.出现了死锁。
本方唤醒了本方,导致了所有的线程都等待了。
解决方式就是:唤醒所有等待的线程。这样既唤醒了本方也唤醒对方。
虽然解决了多生产消费的问题,但是有些低效。
解决方法一: 唤醒所有等待的线程
class Resource{
private String name;//定义一个商品都有名字。
private int count = 1; //定义一个商品的编号。
private boolean flag = false;//定义用来判断是否有商品的标记。
public synchronized void set(String name){
while(flag)
try{wait();}catch(Exception e){}//t1,t2
this.name = name+"--"+count;//蛋糕1 蛋糕2 蛋糕3
count++;
System.out.println(Thread.currentThread().getName()+"---生产了,"+this.name);
//生产蛋糕1 生产蛋糕2 生产蛋糕3
flag = true; //将标记改为true。
notifyAll();//唤醒等待的线程。
}
public synchronized void get()//{
while(!flag)
try{wait();}catch(Exception e){}//t3.t4
System.out.println(Thread.currentThread().getName()+"-------消费了....."+this.name);//消费蛋糕1
flag = false;
notifyAll();
}
}
//定义生产者的任务。
class Producer implements Runnable{
private Resource r;
Producer(Resource r){
this.r = r;
}
public void run(){
while(true){
r.set("蛋糕");
}
}
}
//定义消费者的任务。
class Consumer implements Runnable{
private Resource r;
Consumer(Resource r){
this.r = r;
}
public void run(){
while(true){
r.get();
}
}
}
class ThreadDemo_Producer_Consumer2 {
public static void main(String[] args) {
Resource r = new Resource();
Producer pro = new Producer(r);
Consumer con = new Consumer(r);
//创建两个生产者线程。
Thread t1 = new Thread(pro);
Thread t2 = new Thread(pro);
//创建两个消费者线程。
Thread t3 = new Thread(con);
Thread t4 = new Thread(con);
t1.start();
t2.start();
t3.start();
t4.start();
}
}解决方法二:使用Lock锁对象和Condition监听对象
根据jdk1.5 版本的特性,创建一个锁对象。比同步的隐式锁操作要更加的面向对象。提供了显示的锁操作。
通过lock锁获取监视器方法对象。Condition 负责生产者的监视操作。
再创建一个监视器方法对象。负责消费者的监视操作。
解决了低效的问题
import java.util.concurrent.locks.*;
class Resource{
//定义一个商品都有名字。
private String name;
//定义一个商品的编号。
private int count = 1;
//定义用来判断是否有商品的标记。
private boolean flag = false;
//根据jdk1.5 版本的特性,创建一个锁对象。比同步的隐式锁操作要更加的面向对象。提供了显示的锁操作。
final Lock lock = new ReentrantLock();//互斥锁。
//通过lock锁获取监视器方法对象。Condition 负责生产者的监视操作。
final Condition producer = lock.newCondition();
//再创建一个监视器方法对象。负责消费者的监视操作。
final Condition consumer = lock.newCondition();
public void set(String name)//{
//通过锁对象进行显示的获取锁操作。
lock.lock();//获取锁,锁的开始位置,想从哪里锁就把锁放在什么地方
try{
while(flag)
try{producer.await();}catch(Exception e){}//t1,t2
this.name = name+"--"+count;
count++;
System.out.println(Thread.currentThread().getName()+"生产了,"+this.name);
//将标记改为true。
flag = true;
consumer.signal();//唤醒等待的线程。
}finally{
//释放锁。
lock.unlock();//定义在finally中,要求一定被释放。
}
}
public void get()//{
lock.lock();
try{
while(!flag)
try{consumer.await();}catch(Exception e){}//t3.t4
System.out.println(Thread.currentThread().getName()+"--消费了."+this.name); //消费蛋糕1
flag = false;
producer.signal();
}finally{
lock.unlock();
}
}
}
//定义生产者的任务。
class Producer implements Runnable{
private Resource r;
Producer(Resource r){
this.r = r;
}
public void run(){
while(true){
r.set("蛋糕");
}
}
}
//定义消费者的任务。
class Consumer implements Runnable{
private Resource r;
Consumer(Resource r){
this.r = r;
}
public void run(){
while(true){
r.get();
}
}
}
class ThreadDemo_Producer_Consumer3 {
public static void main(String[] args) {
Resource r = new Resource();
Producer pro = new Producer(r);
Consumer con = new Consumer(r);
//创建两个生产者线程。
Thread t1 = new Thread(pro);
Thread t2 = new Thread(pro);
//创建两个消费者线程。
Thread t3 = new Thread(con);
Thread t4 = new Thread(con);
t1.start();
t2.start();
t3.start();
t4.start();
}
}标签:多线程
原文地址:http://8477424.blog.51cto.com/8467424/1774102