前言
在阅读本篇文章之前建议先看看,[Java内存模型和volatile、synchronized] (https://www.jianshu.com/p/ad43ac25831a)中介绍过的Java虚拟机对每个对象实现的monitor锁。
wait
- 调用wait方法能将当前线程阻塞掉,直到调用notify或者notifyAll方法来让线程重新去尝试获取锁(即唤醒线程)。
- wait方法是一个本地方法,它是通过一个monitor对象锁来实现的,只有拥有了该对象的监视器锁才能调用wait方法,那么怎么调用wait方法呢?
- 是通过增加synchronized关键字来实现的,这也是为什么wait必须在synchronized修饰的代码中运行的原因。但只要调用了wait方法,monitor锁就会被马上释放掉。
notify
与wait同理也必须在有synchronized关键字修饰的代码中才能够调用,同样都是当前对象调用。
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()
//或者wait()
}
- sleep必须捕获异常,而wait,notify和notifyAll不需要捕获异常