wait,notify,sleep,yield,join方法的实现

前言

在阅读本篇文章之前建议先看看,[Java内存模型和volatile、synchronized] (www.jianshu.com/p/ad43ac258…)中介绍过的Java虚拟机对每个对象实现的monitor锁。

wait

  1. 调用wait方法能将当前线程阻塞掉,直到调用notify或者notifyAll方法来让线程重新去尝试获取锁(即唤醒线程)。
  2. wait方法是一个本地方法,它是通过一个monitor对象锁来实现的,只有拥有了该对象的监视器锁才能调用wait方法,那么怎么调用wait方法呢?
  3. 是通过增加synchronized关键字来实现的,这也是为什么wait必须在synchronized修饰的代码中运行的原因。但只要调用了wait方法,monitor锁就会被马上释放掉。

notify

与wait同理也必须在有synchronized关键字修饰的代码中才能够调用,同样都是当前对象调用。

sleep

  1. 使用sleep方法会让线程暂停,也需要synchronized关键字,但只是让出了CPU的时间片,并没有释放掉monitor锁。
  2. 调用sleep的线程是顺序进入的,当前一个线程还没有执行完的时候,后面一个线程是不可能执行同步方法的。
  3. 当wait方法不同,因为它调用后就马上释放掉monitor锁了,所以其他线程就能够获取到锁,执行当前对象的调用。当之前调用wait方法的线程被唤醒后,才能继续是参与锁的竞争(注意:这里并不是说能马上获取到锁,是获取到锁之后才能进行执行)。

yield

  1. 当一个线程正在执行的时候,它拥有最高的优先级。
  2. 当调用yield方法时,JVM可能会把运行机会给其他拥有相同优先级的线程,也可能不会因为yield只是提出个建议并没有强制切换线程。
  3. 当切换线程后,当前调用yield的线程进行到可运行状态,而不是等待或阻塞状态。

join

阻塞当前正在执行的进程,执行调用join方法的线程,执行完毕后,再去唤醒当前线程。

补充:为什么wait方法在Object类中,Sleep在Thread类中?

这两者的施加者是有本质区别的.

  1. sleep()是让某个线程暂停运行一段时间,其控制范围是由当前线程决定,也就是说,在线程里面决定.
  2. wait()是由某个确定的对象来调用的
  • 两者都可以让线程暂停一段时间,但是本质的区别是一个线程的运行状态控制,一个是线程之间的通讯的问题

sleep和wait之间区别有:

  1. 这两个方法来自不同的类分别是Thread和Object
  2. 最主要是sleep方法没有释放锁,而wait方法释放了锁,使得其他线程可以使用同步控制块或者方法。
  3. wait,notify和notifyAll只能在同步控制方法或者同步控制块里面使用,而sleep可以在任何地方使用
   synchronized(x){
      x.notify()
     //或者wait()
   }
复制代码
  1. sleep必须捕获异常,而wait,notify和notifyAll不需要捕获异常

猜你喜欢

转载自juejin.im/post/5d6a0675f265da03bf0f5999