线程通信-wait和notify方法介绍:
java.lang.Object类提供类两类用于操作线程通信的方法.
wait():执行该方法的线程对象释放同步锁,JVM把该线程存放到等待池中,等待其他的线程唤醒该线程.
notify:执行该方法的线程唤醒在等待池中等待的任意一个线程,把线程转到锁池中等待.
notifyAll():执行该方法的线程唤醒在等待池中等待的所有的线程,把线程转到锁池中等待.
注意:上述方法只能被同步监听锁对象来调用,否则报错IllegalMonitorStateException…
假设A线程和B线程共同操作一个X对象(同步锁),A,B线程可以通过X对象的wait和notify方法来进行通信,流程如下:
1:当A线程执行X对象的同步方法时,A线程持有X对象的锁,B线程没有执行机会,B线程在X对象的锁池中等待.
2:A线程在同步方法中执行X.wait()方法时,A线程释放X对象的锁,进入A线程进入X对象的等待池中.
3:在X对象的锁池中等待锁的B线程获取X对象的锁,执行X的另一个同步方法.
4:B线程在同步方法中执行X.notify()方法时,JVM把A线程从X对象的等待池中移动到X对象的锁池中,等待获取锁.
5:B线程执行完同步方法,释放锁.A线程获得锁,继续执行同步方法.
————————
wait() 导致当前线程等待,直到其他线程调用此对象的 notify()方法或 notifyAll()方法前,导致当前线程等待。
notify() 唤醒在此同步监视器上等待的单个线程,如果有多个线程在此同步监视器上等待,则会唤醒其中的一个,
功能代码
public class RoomMate implements Runnable{
Object cesuo=new Object();//临界资源厕所对象
public void toBrush(){
System.out.println(Thread.currentThread().getName()+"进去刷牙");
try {
Thread.sleep(2000);
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
System.out.println(Thread.currentThread().getName()+"刷完牙出来了");
}
public void toWC(){
System.out.println(Thread.currentThread().getName()+"进去上厕所");
try {
Thread.sleep(2000);
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
System.out.println(Thread.currentThread().getName()+"上完厕所出来了");
}
@Override
public void run() {
// TODO Auto-generated method stub
synchronized (cesuo) {
if("张三".equals(Thread.currentThread().getName())){
this.toBrush();
try {
cesuo.wait();//释放锁,张三进入等待状态
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
this.toWC();
cesuo.notify();//唤醒李四
}else{
this.toBrush();
cesuo.notify();//唤醒张三
try {
cesuo.wait();//李四等待
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
this.toWC();
cesuo.notify();//唤醒李四
}
}
}
}
测试
public class Demo {
public static void main(String[] args) {
Runnable r=new RoomMate();
Thread t1=new Thread(r, "张三");
Thread t2=new Thread(r, "李四");
t1.start();
t2.start();
}
}