JDK在Object对象中提供了2个非常重要的接口线程方法wait方法和notify方法,也就是所有Java对象都有这2个方法,意思就是当在一个实例Java对象上调用wait方法,那么当前线程就会从执行状态转变成等待状态,同时释放在实例对象上的锁,直到其它线程在刚才那个实例对象上调用notify方法并且释放实例对象上的锁,那么刚才那个当前线程才会再次获取实例对象锁并且继续执行。这样我们通过object对象就可以让多线程之间进行有效通信。
那么这2个方法是如何工作的呢?比如我们有一个person对象,如果一个线程T1调用person.wait(),那么这个线程a就会进入person对象的等待队列,在这个等待队列中可能还有线程T2,线程T3,线程T4,因为系统可能通过4个线程来等待person实例对象,当我们调用person.notify()方法,它就会从这个等待队列中随机选一个线程,并将其唤醒,在这里这个选择是不公平的,也就是选择线程T1,T2,T3,T4是随机的,当然了也可以调用person,notifyAll()方法,这个方法会把线程T1,T2,T3,T4全部唤醒。
需要注意的是person.wait()方法并不是随便调用的,它必须包含在对应的synchronzied中,无论是wait()方法还是notify()方法都需要首先获取目标对象上的一个监视器,如下图所示:
我们看下面的代码,注意打印结果,就能看到程序运行情况,需要注意的是Object.wait()和Thread.sleep()都可以让程序等待若干时间,区别在于sleep()方法不会释放对象锁,而wait会释放锁
package me.lishuo;
/**
* @date 2018/7/21 下午5:10
*/
public class Demo{
final static Object person =new Object();
public static class T1 extends Thread{
public void run(){
synchronized (person){
System.out.println(System.currentTimeMillis()+"T1 come");
try{
System.out.println(System.currentTimeMillis()+"T1 wait");
person.wait();
}catch (InterruptedException r){
r.getStackTrace();
}
System.out.println(System.currentTimeMillis()+"T1 over");
}
}
}
public static class T2 extends Thread{
public void run(){
synchronized (person){
System.out.println(System.currentTimeMillis()+"T2 come");
person.notify();
System.out.println(System.currentTimeMillis()+"T2 over");
try{
Thread.sleep(2000);
}catch (InterruptedException r){
r.getStackTrace();
}
}
}
}
public static void main(String args[]){
try{
Thread thread1=new T1();
Thread thread2=new T2();
thread1.start();
thread2.start();
}catch (Exception e){
e.printStackTrace();
}
}
}
关于线程的状态参考https://blog.csdn.net/xingjing1226/article/details/81977129