目录
因为线程之间是抢占式执行的,所以线程之间执行的先后顺序难以预知。但是实际开发中,我们希望线程之间的执行顺序是能被掌控的,比如线程2开始之前,需要线程1的某个任务先被执行。也就是说,很多时候,我们需要线程之间能协调工作,这就主要涉及到以下三个方法(均为Object类的方法):
- wait():让当前线程进行等待状态
- notify():唤醒在当前状态上等待的线程(只能唤醒一个等待该对象的对象锁的线程)
- notifyAll():唤醒在当前状态上等待的线程(能唤醒所有等待该对象的对象锁的线程)
1.wait()方法
wait方法的作用是使当前线程进入阻塞等待.
当代码执行到wait()方法时,线程会经过以下过程:
- 当前线程释放锁.
- 使当前执行代码的线程进行等待(把线程放到等待队列中).
- 满足一定条件时线程被唤醒,重新尝试获取这个锁.
注意:wait()必须要和synchronized一起使用.因为wait()方法一执行就要先释放锁,所以wait必须要放到synchronized中使用.且synchronized锁的对象是和调用wait方法的对象是同一个对象.
wait使用示例:
运行这段代码:
发现代码就停在这里了,这时我们需要使用另外一个唤醒的方法notify().让其能继续进行下去.
wait 结束等待的条件:
- 其他线程调用该对象的notify方法
- 其他线程调用该等待线程的interrupted方法,导致wait抛出InterruptedException异常
- wait等待的时间超时(wait方法重载了一个带有timeout参数的方法,来指定等待时间)
2. notify方法
notify方法的作用是唤醒等待的线程.
方法notify()要在同步方法或同步方法块中调用,该方法是用来通知那些正在等待该对象的对象锁的其他线程,对其发出通知notify,并使他们重新获取该对象的对象锁.
如果有多个线程都在等待,则线程调度器会随机挑选出一个呈wait状态的线程,没有先来后到这一说.
在notify方法后,当前线程不会马上释放该对象锁,要等到执行notify()方法的线程将程序执行完,也就是退出同步代码块之后才会释放对象锁.
使用notify方法唤醒线程:
分析以上代码,线程t1执行到locker.wait()时释放锁,阻塞等待;t2线程拿到锁,执行到locker.notify()时,将t1线程唤醒,但是此时t2线程还不会先将锁释放,会先将同步代码块执行完才会释放锁,之后t1线程再次开始执行.运行结果如下:
wait和notify方法要搭配使用,比如实现一个场景:线程1与线程2并发执行,线程1执行到某一条件时需要拿到线程2的结果才能继续往下执行,这时就可以用wait方法使其进入等待,当线程2将结果计算出来后,可用notify方法将等待该对象的对象锁的某一线程唤醒,使其重新拿到锁,继续执行.
notifyAll()方法:
notify方法只是唤醒一个线程,而notifyAll()方法可以一次唤醒所有的等待线程,比如有3个线程都在等待该对象的对象锁,那么notifyAll()方法可将这3个线程全部唤醒,但是这3个线程仍需要竞争这一把锁.
wait VS sleep
wait需要搭配synchronized使用,而sleep不需要;
wait是Object的方法,sleep是Thread的静态方法.
二者其实没有任何可比性而言.wait是用于线程之间的通信的,而sleep是让线程阻塞一段时间的.唯一的相同点就是都可以让线程停止执行一段时间.