【多线程:锁】生产者消费者
01.介绍
例子介绍:
这里的例子是生产者消费者模型
我们有一个生产糖果的生产者 与 一个消费糖果的消费者,假设 生产是一个线程 消费是一个线程 总共的糖果池是6,这时会出现两个错误 第一如果我们的消费者的消费速度大于生产者的生产速度 就会出现 把糖果消费到0的时候 生产者还没有生产 但消费者还在消费 导致错误,以及生产速度大于消费速度 就会导致 生产到了6个 但是消费者还没有消费 生产者就再次消费 导致糖果池大于了6个 导致错误。
这个例子说明了什么问题:
这个例子说明了 当两个线程出现类似生产消费等这种关系时 极有可能会出现 某一个线程已经开始不满足接着运行的条件了 但是它还是霸占着cpu 而不是给另外的线程运行,所以我们要解决这个问题,对于上述例子来说,解决方法就是 当生产者生产糖果达到6时就切换到消费线程,当消费线程消费到0时就切换到生产线程。
如何解决:
在多线程中 我们有两个方法,一个是 wait方法 用于使当前线程陷入等待状态,一个是 notify方法 用于唤醒其他线程。在上述例子中 如果生产速度大于消费速度时 我们就可以在 生产线程 达到糖果为6这个条件时 进行wait使其陷入等待 并使用notify方法 把消费线程唤醒(如果消费线程本来就没有wait就不需要唤醒),之后就可以切换到消费线程运行 直到消费线程消费后使得生产线程不满足达到6个糖果的条件 我们再使用notify唤醒 生产线程。一直这样下去保证了程序可以正常运行。
02.wait、sleep、notify、notifyAll方法的介绍
wait
wait方法是线程等待,这个方法会使拥有对象锁的线程进入等待状态,直到其他线程调用notify或notifyAll方法才能“唤醒”
sleep
sleep方法是线程暂停,是线程用来控制自身流程的,它会使线程暂停一段时间,把执行机会让给其他线程,等计时时间一到,此线程会自动“苏醒”
notify
notify是用来用来唤醒调用wait方法进入等待锁资源队列的线程,唤醒正在等待此对象监视器的单个线程。 如果有多个线程在等待,则选择其中一个随机唤醒(由调度器决定),唤醒的线程享有公平竞争资源的权利
notifyAll
notifyAll是用来用来唤醒调用wait方法进入等待锁资源队列的线程,唤醒正在等待此对象监视器的所有线程,唤醒的所有线程公平竞争资源
原版
class SynStack
{
private char[] data = new char[6];
private int cnt = 0;
public void push(char ch)
{
data[cnt]=ch;
++cnt;
System.out.println("1111");
}
public char pop()
{
--cnt;
System.out.printf("2222");
return data[cnt];
}
}
class Producer implements Runnable
{
private SynStack ss=null;
public Producer(SynStack ss)
{
this.ss=ss;
}
public void run()// throws Exception //错误 Runnable里没有异常处理
{
// push('a');//错误
try
{
Thread.sleep(2000);
}
catch(Exception e)
{
}
ss.push('a');
}
}
class Consumer implements Runnable
{
private SynStack ss=null;
public Consumer(SynStack ss)
{
this.ss=ss;
}
public void run()
{
System.out.printf("%c\n",ss.pop());
}
}
public class TestPC
{
public static void main(String[] args)
{
SynStack ss = new SynStack();
Producer p = new Producer(ss);
Consumer c = new Consumer(ss);
Thread t1=new Thread(p);
Thread t2=new Thread(c);
t1.start();
t2.start();//错误 因为无法判断是先进还是先出,如果先进则没错 但如果先出 则现在栈中没有元素 导致数组下标越界
}
}
结果
报数组溢出错误
解释:
我们观察程序,发现我们创造了一个栈 有两个方法 分别是pop()与push() 出栈与入栈,这个栈其实就是 糖果池 而糖果池的最大容量就是 这个栈的最大值6之后我们创建了生产者与消费者且他们都实现了Runnable接口与run方法,生产者会在run方法中push()一个元素 也就是增加一个糖果,消费者会在run方法中pop()一个元素 也就是减少一个糖果。
最后我们创建了生产线程与消费线程,我们令生产线程的生产速率小于消费线程,最终会发现当糖果池为0后消费线程还在消费,导致溢出错误。
解决方案
class SynStack
{
private char[] data = new char[6];
private int cnt = 0;
public synchronized void push(char ch)
{
while(cnt==data.length)
//注意:
//这里用while是为了保证 栈为满而暂停,然后下一次如果再满还是再这个循环里 保证再暂停,
//如果改成if 则只能暂停一次 下一次从暂停的那个位置继续走
//但如果是同步过以后则可以用 if,因为如果满 则暂停(wait)生产线程 同时解除霸占,转换到
//消费线程 进行消费操作 并锁住消费线程对象 导致一定会运行完消费操作 使得栈不为满 之后
//唤醒生产操作 并结束消费程序 解锁,之后再竞争
// 但这种while与if都可以用的情况 仅在 只有两个相互依赖的线程 的条件下成立,我们可以想一想 如果我们现在来一个毫不相干的线程c c运行完后也
// 会唤醒某个线程 这样不满足条件的线程依旧可能被唤醒 并且因为是if 所以直接跳出if进行生产操作 导致错误
//使用wait会解锁,sleep则不会
{
try
{
this.wait();
}
catch(Exception e)
{}
}
this.notify();//唤醒其他某一个暂停的线程 对于这个例子 就是唤醒消费线程
data[cnt]=ch;
++cnt;
System.out.printf("生产线程正在生产第%d个产品,该产品是:%c\n",cnt,ch);
}
public synchronized char pop()
{
char ch;
if(cnt==0)
//注意:这里的注意同上
{
try
{
this.wait();
}
catch(Exception e)
{}
}
this.notify();//唤醒其他某一个暂停的线程 对于这个例子 就是唤醒生产线程
ch=data[cnt-1];
System.out.printf("消费线程正在消费第%d个产品,该产品是:%c\n",cnt,ch);
cnt--;
return ch;
}
}
class Producer implements Runnable
{
private SynStack ss=null;
public Producer(SynStack ss)
{
this.ss=ss;
}
public void run()// throws Exception //错误 Runnable里没有异常处理
{
// push('a');//错误
char ch;
for(int i=0;i<20;i++)
{
try
{
Thread.sleep(200);//这种是生产比较慢 消费快
}
catch(Exception e)
{
}
ch=(char)('a'+i);
ss.push(ch);
}
}
}
class Consumer implements Runnable
{
private SynStack ss=null;
public Consumer(SynStack ss)
{
this.ss=ss;
}
public void run()
{
for(int i=0;i<20;i++)
{
/*try
{
Thread.sleep(200);//这种是消费比较慢 生产快
}
catch(Exception e)
{
}*/
ss.pop();
}
}
}
public class TestPC_2
{
public static void main(String[] args)
{
SynStack ss = new SynStack();
Producer p = new Producer(ss);
Consumer c = new Consumer(ss);
Thread t1=new Thread(p);
Thread t2=new Thread(c);
t1.start();
t2.start();
}
}
结果
生产线程正在生产第1个产品,该产品是:a
消费线程正在消费第1个产品,该产品是:a
生产线程正在生产第1个产品,该产品是:b
消费线程正在消费第1个产品,该产品是:b
生产线程正在生产第1个产品,该产品是:c
消费线程正在消费第1个产品,该产品是:c
生产线程正在生产第1个产品,该产品是:d
消费线程正在消费第1个产品,该产品是:d
生产线程正在生产第1个产品,该产品是:e
消费线程正在消费第1个产品,该产品是:e
生产线程正在生产第1个产品,该产品是:f
消费线程正在消费第1个产品,该产品是:f
生产线程正在生产第1个产品,该产品是:g
消费线程正在消费第1个产品,该产品是:g
生产线程正在生产第1个产品,该产品是:h
消费线程正在消费第1个产品,该产品是:h
生产线程正在生产第1个产品,该产品是:i
消费线程正在消费第1个产品,该产品是:i
生产线程正在生产第1个产品,该产品是:j
消费线程正在消费第1个产品,该产品是:j
生产线程正在生产第1个产品,该产品是:k
消费线程正在消费第1个产品,该产品是:k
生产线程正在生产第1个产品,该产品是:l
消费线程正在消费第1个产品,该产品是:l
生产线程正在生产第1个产品,该产品是:m
消费线程正在消费第1个产品,该产品是:m
生产线程正在生产第1个产品,该产品是:n
消费线程正在消费第1个产品,该产品是:n
生产线程正在生产第1个产品,该产品是:o
消费线程正在消费第1个产品,该产品是:o
生产线程正在生产第1个产品,该产品是:p
消费线程正在消费第1个产品,该产品是:p
生产线程正在生产第1个产品,该产品是:q
消费线程正在消费第1个产品,该产品是:q
生产线程正在生产第1个产品,该产品是:r
消费线程正在消费第1个产品,该产品是:r
生产线程正在生产第1个产品,该产品是:s
消费线程正在消费第1个产品,该产品是:s
生产线程正在生产第1个产品,该产品是:t
消费线程正在消费第1个产品,该产品是:t
解释
因为消费线程比较快,所以现在的情况就是生产一个立马就被消费了