1.实现线程通信例子
第一步创建资源类,在资源类中创属性和操作方法
第二步在资源类操作方法
(1)判断
(2)干活
(3)通知
第三步创建多个线程调用资源类的操作方法
例子
有两个线程,实现对一个初始值是0的变量,其中一个线程实现+1,另外一个线程-1,让这个变量一直在 0 ,1中进行徘徊
在+1的操作中判断变量是否是0如果不是调用object中的wait方法让其进行等待,然后让另外一个线程进行-1之后使用object.notify方法进行+1线程的唤醒操作
例子代码
//第一步创建资源类,定义属性和操作方法
class Share{
//初始值
private int number = 0;
//+1的方法
public synchronized void incr() throws InterruptedException {
//第二步 判断干活和通知
if(number != 0){
//判断number值是否是0,如果不是0,等待
this.wait();
}
//如果number值是0,就+1操作
number++;
System.out.println(Thread.currentThread().getName()+"::"+number);
//通知其他线程
this.notifyAll();
}
//-1的方法
public synchronized void decr() throws InterruptedException {
//判断
if(number != 1){
this.wait();
}
//干活
number--;
System.out.println(Thread.currentThread().getName()+"::"+number);
//通知其他的线程
this.notifyAll();
}
}
public class ThreadDemo1 {
//第三步创建多个线程,调用资源类的操作方法
public static void main(String[] args) {
Share share = new Share();
//创建两个线程
new Thread(()->{
for (int i = 0; i<= 10;i++){
try {
share.incr();
} catch (InterruptedException e) {
e.printStackTrace();
}
}
},"aa").start();
new Thread(()->{
for (int i = 0; i<= 10;i++){
try {
share.decr();
} catch (InterruptedException e) {
e.printStackTrace();
}
}
},"bb").start();
}
}
2.虚假唤醒问题
这里我们进行多个线程进行操作 线程aa +1 ,bb -1, cc +1 ,dd-1,四个线程进行操作
然后我们进行测试,我们发现这里面出现了-1。
为什么会出现-1?
这里我们要从wait方法进行分析,wait有一个特点就是从那里睡,就会从哪里醒。
因此出现这种问题,是因为aa 进行+1后使用了notifyAll进行其他线程的唤醒,假如bb抢到了资源的使用权,之后bb进行判断当前值为1进行-1 ,值变为0,然后进行notifyAll进行其他线程的唤醒,假如现在dd抢到了资源使用权,进行判断变量值为0,然后进行wait,wait会唤醒其他线程,然后bb又抢到了,bb进行wait ,然后dd抢到了使用权,在之前wait的睡下去的地方醒过来,继续走下去进行了-1的操作,就导致了出现-1的问题这就是虚假唤醒。
3.如何解决虚假唤醒的问题?
如上图,当我们使用了,wait的方法时,应该判断使用while循环来进行判断,即可解决虚假唤醒的问题