Java中wait sleep await 区别于联系

前言:本文解决的问题

  • wait() await() sleep()这三个方法有申请区别

在找工作的各种笔试题目中,经常看到wait()、sleep()还有await(),功能都很相似,到底有什么区别?什么时候该用哪一种方法

1. wait() VS sleep()

wait和sleep的比较可以说是高频面试题。方法原型分别为:

 public final native void wait(long timeout) throws InterruptedException;

 public static native void sleep(long millis) throws InterruptedException;

同:

  • 都是线程同步时会用到的方法,使当前线程暂停运行,把运行机会交给其它线程。
  • 如果任何线程在等待期间被中断都会抛出InterruptedException
  • 都是native方法

异:

  • 所在类不同,wait()是Object超类中的方法;而sleep()是线程Thread类中的方法
  • 关键点是对锁的保持不同,wait会释放锁;而sleep()并不释放锁
  • 唤醒方法不完全相同,wait依靠notify或者notifyAll、中断发生、或者到达指定时间来唤醒;而sleep()则是到达指定的时间后被唤醒。
  • 使用的位置不同,wait只能用在同步代码块中,而sleep用在任何位置。

2. wait() VS await()

这两个长得很像。await()的实现比较复杂。

  public final void await() throws InterruptedException {
            if (Thread.interrupted())
                throw new InterruptedException();
            Node node = addConditionWaiter();
            int savedState = fullyRelease(node);
            int interruptMode = 0;
            while (!isOnSyncQueue(node)) {
                LockSupport.park(this);
                if ((interruptMode = checkInterruptWhileWaiting(node)) != 0)
                    break;
            }
            if (acquireQueued(node, savedState) && interruptMode != THROW_IE)
                interruptMode = REINTERRUPT;
            if (node.nextWaiter != null) // clean up if cancelled
                unlinkCancelledWaiters();
            if (interruptMode != 0)
                reportInterruptAfterWait(interruptMode);
        }

先说下来源,await是ConditionObject类里面的方法,ConditionObject实现了Condition接口;而ReentrantLock里面默认有实现newCondition()方法,新建一个条件对象。该方法就是用在ReentrantLock中根据条件来设置等待。唤醒方法也是由专门的Signal()或者Signal()来执行。另外await会导致当前线程被阻塞,会放弃锁,这点和wait是一样的。

由于所在的超类不同使用场景也不同,wait一般用于Synchronized中,而await只能用于ReentrantLock锁中,具体如下
wait()

          synchronized (obj) {
              while (<condition does not hold>)
               obj.wait(timeout);
             ... // Perform action appropriate to condition
         }

await()主要见上文。

3 notify signal

顺便说下这二者的区别,notify使用来唤醒使用wait的线程;而signal是用来唤醒await线程。

猜你喜欢

转载自www.cnblogs.com/java-learner/p/9652027.html