前言
在阅读本篇文章以前建议先看看,[Java内存模型和volatile、synchronized] (www.jianshu.com/p/ad43ac258…)中介绍过的Java虚拟机对每一个对象实现的monitor锁。java
wait
- 调用wait方法能将当前线程阻塞掉,直到调用notify或者notifyAll方法来让线程从新去尝试获取锁(即唤醒线程)。
- wait方法是一个本地方法,它是经过一个monitor对象锁来实现的,只有拥有了该对象的监视器锁才能调用wait方法,那么怎么调用wait方法呢?
- 是经过增长synchronized关键字来实现的,这也是为何wait必须在synchronized修饰的代码中运行的缘由。但只要调用了wait方法,monitor锁就会被立刻释放掉。
notify
与wait同理也必须在有synchronized关键字修饰的代码中才可以调用,一样都是当前对象调用。spa
sleep
- 使用sleep方法会让线程暂停,也须要synchronized关键字,但只是让出了CPU的时间片,并无释放掉monitor锁。
- 调用sleep的线程是顺序进入的,当前一个线程尚未执行完的时候,后面一个线程是不可能执行同步方法的。
- 当wait方法不一样,由于它调用后就立刻释放掉monitor锁了,因此其余线程就可以获取到锁,执行当前对象的调用。当以前调用wait方法的线程被唤醒后,才能继续是参与锁的竞争(注意:这里并非说能立刻获取到锁,是获取到锁以后才能进行执行)。
yield
- 当一个线程正在执行的时候,它拥有最高的优先级。
- 当调用yield方法时,JVM可能会把运行机会给其余拥有相同优先级的线程,也可能不会由于yield只是提出个建议并无强制切换线程。
- 当切换线程后,当前调用yield的线程进行到可运行状态,而不是等待或阻塞状态。
join
阻塞当前正在执行的进程,执行调用join方法的线程,执行完毕后,再去唤醒当前线程。线程
补充:为何wait方法在Object类中,Sleep在Thread类中?
这二者的施加者是有本质区别的.
- sleep()是让某个线程暂停运行一段时间,其控制范围是由当前线程决定,也就是说,在线程里面决定.
- wait()是由某个肯定的对象来调用的
- 二者均可以让线程暂停一段时间,可是本质的区别是一个线程的运行状态控制,一个是线程之间的通信的问题
sleep和wait之间区别有:
- 这两个方法来自不一样的类分别是Thread和Object
- 最主要是sleep方法没有释放锁,而wait方法释放了锁,使得其余线程可使用同步控制块或者方法。
- wait,notify和notifyAll只能在同步控制方法或者同步控制块里面使用,而sleep能够在任何地方使用
synchronized(x){
x.notify()
}
复制代码
- sleep必须捕获异常,而wait,notify和notifyAll不须要捕获异常