wait() notify() notifyAll() 的使用
package thread;
class printer implements Runnable {
private int i = 1;
@Override
public void run() {
while(true){
synchronized (this){
notify();
System.out.println(Thread.currentThread().getName() + ":" + i);
++i;
if(i>100){
break;
}
try {
wait();
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
}
}
public class communication {
public static void main(String[] args){
printer p = new printer();
Thread t1 = new Thread(p);
Thread t2 = new Thread(p);
t1.start();
t2.start();
}
}
三个方法:
- wait() 线程一旦执行该方法,就会进入阻塞状态,并且释放锁(同步监视器)。
- notify() 线程一旦执行该方法,就会唤醒其他被wait的线程,如果有多个线程被wait,则唤醒优先级最高的那个线程。
- notifyAll() 线程一旦执行该方法,就会唤醒其他所有被wait的线程。
- 这三个方法属于线程通信,必须写在synchronized代码块或者synchronized方法中,不能写在同步锁中。
上面代码实现交替打印原理:
- 假如最开始线程1握到锁进入同步代码块中,执行notify就会唤醒线程2等待抢锁,然后执行wait后线程1等待并抛锁,此时线程2就开始执行。
说明:
- 上程序中的wait()和notify()都没有写是谁调用的方法,是因为是this调用的。原因是wait() notify() notifyAll()三个方法都是Object类的方法,而Object类是所有类的父类,所以所有类都可以调用这三个方法
- wait() notify()必须通过同步监视器调用,上面程序锁是this,所以调用wait() notify()是用this调用的(可省略)
- 但是如果锁换成其他对象例如p,那么就要用p.wait() p.notify()
sleep()和wait()的区别:
- 相同点:一旦执行,该线程都进入到阻塞状态
- 声明位置的不同:sleep在Thread类中声明,wait在Object类中声明
- 使用位置不同:sleep在任何位置都可以,wait只能在同步代码块和同步方法中使用
- wait执行后会释放锁,sleep不会