object.wait()和notify()

1,简介

wait和notify是object的方法,也就是说所有对象都有这两个方法,这两个方法可以用来阻塞当前线程(同时放弃互斥锁)或者是 唤醒其他调用wait方法陷入阻塞的线程(不能唤醒那些因为抢占锁而阻塞的队列)

代码样例


public class NotifyAndWaitT {

   public static void main(String[] args) throws InterruptedException {


       Thread waitThread = new Thread(() -> {

           synchronized (NotifyAndWaitT.class) {
               System.out.println("wait get the lock");
               try {
                   NotifyAndWaitT.class.wait();
                   System.out.println("wait wake up");
               } catch (InterruptedException e) {
                   e.printStackTrace();
               }
           }
       });

       Thread notifyThread = new Thread(() -> {
           try {
               Thread.currentThread().sleep(100);
           } catch (InterruptedException e) {
               e.printStackTrace();
           }

           synchronized (NotifyAndWaitT.class) {
               System.out.println("notify get the lock");
               try {
                   NotifyAndWaitT.class.notify();
                   Thread.currentThread().sleep(1000);
                   System.out.println("notify end---------");
               } catch (InterruptedException e) {
                   e.printStackTrace();
               }
           }
       });

       waitThread.start();
       notifyThread.start();
       waitThread.join();
       notifyThread.join();
       System.out.println("main end*******************");
   }


}

2. wait 运行过程

能够执行wait和notify的前提是代码已经进入了synchronized包含的代码块中。

当然synchronized现在通过优化,增加了偏向锁,轻量级锁,但是在执行obj.wait()的时候都会膨胀成重量级锁ObjectMonitor (关于每个锁的关联数据可以看看),然后做一系列的操作。

具体看一下执行过程:

参考 object.wait的源码解析.md 可以看到代码执行的逻辑,下面会具体分析一下

首先是每个了解当前线程持有的锁(objectMonitor)拥有的数据结构, objectMonitor在jdk8中的数据结构.md

在这里面可以看到在objectMonitor中有一个地方存储的是 _WaitSet ,这个存放的就是因为obj.wait()进入阻塞的线程。

3. 具体从代码看 wait() 操作

看objectMonitor.cpp文件中的 ObjectMonitor::wait(jlong millis, bool interruptible, TRAPS)


// will need to be replicated in complete_exit above    

void ObjectMonitor::wait(jlong millis, bool interruptible, TRAPS) {    

  Thread * const Self = THREAD ;    

  assert(Self->is_Java_thread(), "Must be Java thread!");    

  JavaThread *jt = (JavaThread *)THREAD;    



  DeferredInitialize () ;    



  // Throw IMSX or IEX.    

  CHECK_OWNER();    



  // check for a pending interrupt    检查是否有中断信号,有的话就抛出异常

  if (interruptible && Thread::is_interrupted(Self, true) && !HAS_PENDING_EXCEPTION) {    

    // post monitor waited event.  Note that this is past-tense, we are done waiting.    

    if (JvmtiExport::should_post_monitor_waited()) {    

       // Note: 'false' parameter is passed here because the    

       // wait was not timed out due to thread interrupt.    

       JvmtiExport::post_monitor_waited(jt, this, false);    

    }    

    TEVENT (Wait - Throw IEX) ;    

    THROW(vmSymbols::java_lang_InterruptedException());    

    return ;    

  }    

  TEVENT (Wait) ;    



  assert (Self->_Stalled == 0, "invariant") ;    

  Self->_Stalled = intptr_t(this) ;    

  jt->set_current_waiting_monitor(this);    



  // create a node to be put into the queue    

  // Critically, after we reset() the event but prior to park(), we must check    

  // for a pending interrupt.    

//把当前线程包装成一个 objectwaiter对象,

  ObjectWaiter node(Self);    

  node.TState = ObjectWaiter::TS_WAIT ;    

  Self->_ParkEvent->reset() ;    

  OrderAccess::fence();          // ST into Event; membar ; LD interrupted-flag    



  // Enter the waiting queue, which is a circular doubly linked list in this case    

  // but it could be a priority queue or any data structure.    

  // _WaitSetLock protects the wait queue.  Normally the wait queue is accessed only    

  // by the the owner of the monitor *except* in the case where park()    

  // returns because of a timeout of interrupt.  Contention is exceptionally rare    

  // so we use a simple spin-lock instead of a heavier-weight blocking lock.    

//加锁,把objectwaiter 对象添加到等待的set中



  Thread::SpinAcquire (&_WaitSetLock, "WaitSet - add") ;    

  AddWaiter (&node) ;    

  Thread::SpinRelease (&_WaitSetLock) ;    



  if ((SyncFlags & 4) == 0) {    

     _Responsible = NULL ;    

  }    

  intptr_t save = _recursions; // record the old recursion count    

  _waiters++;                  // increment the number of waiters    

  _recursions = 0;             // set the recursion level to be 1    

  exit (Self) ;                    // exit the monitor    

  guarantee (_owner != Self, "invariant") ;    



  // As soon as the ObjectMonitor's ownership is dropped in the exit()    

  // call above, another thread can enter() the ObjectMonitor, do the    

  // notify(), and exit() the ObjectMonitor. If the other thread's    

  // exit() call chooses this thread as the successor and the unpark()    

  // call happens to occur while this thread is posting a    

  // MONITOR_CONTENDED_EXIT event, then we run the risk of the event    

  // handler using RawMonitors and consuming the unpark().    

  //    

  // To avoid the problem, we re-post the event. This does no harm    

  // even if the original unpark() was not consumed because we are the    

  // chosen successor for this monitor.    

//在这里防止的情况是当前线程刚刚执行完monitor的释放,

//有一个线程进来了,执行完后退出了,又唤醒了当前线程,

//把当前线程从等待队列中给拿出来了,

//但是当前线程实际上还是没有执行park操作

//这样的话再执行下面的park操作的话,

//阻塞了也没有机会唤醒了,不在那个等待条件的队列里面了

// 那么就对当前线程再执行一次unpark,在下面执行park的时候就直接略过了,不会出现问题

  if (node._notified != 0 && _succ == Self) {    

     node._event->unpark();    

  }    



  // The thread is on the WaitSet list - now park() it.    

  // On MP systems it's conceivable that a brief spin before we park    

  // could be profitable.    

  //    

  // TODO-FIXME: change the following logic to a loop of the form    

  //   while (!timeout && !interrupted && _notified == 0) park()    



  int ret = OS_OK ;    

  int WasNotified = 0 ;    

  { // State transition wrappers    

    OSThread* osthread = Self->osthread();    

    OSThreadWaitState osts(osthread, true);    

    {    

      ThreadBlockInVM tbivm(jt);    

      // Thread is in thread_blocked state and oop access is unsafe.    

      jt->set_suspend_equivalent();    

//执行park操作,把当前线程挂起



      if (interruptible && (Thread::is_interrupted(THREAD, false) || HAS_PENDING_EXCEPTION)) {    

          // Intentionally empty    

      } else    

      if (node._notified == 0) {    

        if (millis <= 0) {    

           Self->_ParkEvent->park () ;    

        } else {    

           ret = Self->_ParkEvent->park (millis) ;    

        }    

      }    



      // were we externally suspended while we were waiting?    

      if (ExitSuspendEquivalent (jt)) {    

         // TODO-FIXME: add -- if succ == Self then succ = null.    

         jt->java_suspend_self();    

      }    



    } // Exit thread safepoint: transition _thread_blocked -> _thread_in_vm    





    // Node may be on the WaitSet, the EntryList (or cxq), or in transition    

    // from the WaitSet to the EntryList.    

    // See if we need to remove Node from the WaitSet.    

    // We use double-checked locking to avoid grabbing _WaitSetLock    

    // if the thread is not on the wait queue.    

    //    

    // Note that we don't need a fence before the fetch of TState.    

    // In the worst case we'll fetch a old-stale value of TS_WAIT previously    

    // written by the is thread. (perhaps the fetch might even be satisfied    

    // by a look-aside into the processor's own store buffer, although given    

    // the length of the code path between the prior ST and this load that's    

    // highly unlikely).  If the following LD fetches a stale TS_WAIT value    

    // then we'll acquire the lock and then re-fetch a fresh TState value.    

    // That is, we fail toward safety.    



    if (node.TState == ObjectWaiter::TS_WAIT) {    

        Thread::SpinAcquire (&_WaitSetLock, "WaitSet - unlink") ;    

        if (node.TState == ObjectWaiter::TS_WAIT) {    

           DequeueSpecificWaiter (&node) ;       // unlink from WaitSet    

           assert(node._notified == 0, "invariant");    

           node.TState = ObjectWaiter::TS_RUN ;    

        }    

        Thread::SpinRelease (&_WaitSetLock) ;    

    }    



    // The thread is now either on off-list (TS_RUN),    

    // on the EntryList (TS_ENTER), or on the cxq (TS_CXQ).    

    // The Node's TState variable is stable from the perspective of this thread.    

    // No other threads will asynchronously modify TState.    

    guarantee (node.TState != ObjectWaiter::TS_WAIT, "invariant") ;    

    OrderAccess::loadload() ;    

    if (_succ == Self) _succ = NULL ;    

    WasNotified = node._notified ;    



 



    // post monitor waited event. Note that this is past-tense, we are done waiting.    

    if (JvmtiExport::should_post_monitor_waited()) {    

      JvmtiExport::post_monitor_waited(jt, this, ret == OS_TIMEOUT);    

    }    

    OrderAccess::fence() ;    



    assert (Self->_Stalled != 0, "invariant") ;    

    Self->_Stalled = 0 ;    



    assert (_owner != Self, "invariant") ;    

    ObjectWaiter::TStates v = node.TState ;    

    if (v == ObjectWaiter::TS_RUN) {    

        enter (Self) ;    

    } else {    

        guarantee (v == ObjectWaiter::TS_ENTER || v == ObjectWaiter::TS_CXQ, "invariant") ;    

        ReenterI (Self, &node) ;    

        node.wait_reenter_end(this);    

    }    

    // want residual elements associated with this thread left on any lists.    

    guarantee (node.TState == ObjectWaiter::TS_RUN, "invariant") ;    

    assert    (_owner == Self, "invariant") ;    

    assert    (_succ != Self , "invariant") ;    

  } // OSThreadWaitState()    



  jt->set_current_waiting_monitor(NULL);    



  guarantee (_recursions == 0, "invariant") ;    

  _recursions = save;     // restore the old recursion count    

  _waiters--;             // decrement the number of waiters    



  // Verify a few postconditions    

  assert (_owner == Self       , "invariant") ;    

  assert (_succ  != Self       , "invariant") ;    

  assert (((oop)(object()))->mark() == markOopDesc::encode(this), "invariant") ;    



  if (SyncFlags & 32) {    

     OrderAccess::fence() ;    

  }    



  // check if the notification happened    

//如果线程不是通过notify唤醒的,那就只能是通过 interrupt唤醒的,就抛出异常。

//这也是object.wait()方法可以抛出异常的原因

  if (!WasNotified) {    

    // no, it could be timeout or Thread.interrupt() or both    

    // check for interrupt event, otherwise it is timeout    

    if (interruptible && Thread::is_interrupted(Self, true) && !HAS_PENDING_EXCEPTION) {    

      TEVENT (Wait - throw IEX from epilog) ;    

      THROW(vmSymbols::java_lang_InterruptedException());    

    }    

  }    

}    









1.检查是否有中断,如果有中断的话就抛出中断(返回)

2.把当前线程包装成一个 objectWaiter 对象

3.通过自旋锁把 objectWaiter 添加到 ObjectMonitor 的 _WaitSet 当中

4.释放 objectMonitor 对象

5.如果当前线程处于被 notifyed 状态(针对这种情况出现的可能是下面的解释),对当前线程执行一次 unpark()操作


//在这里防止的情况是当前线程刚刚执行完monitor的释放,

//有一个线程进来了,执行完后退出了,又唤醒了当前线程,

//把当前线程从等待队列中给拿出来了,

//但是当前线程实际上还是没有执行park操作

//这样的话再执行下面的park操作的话,

//阻塞了也没有机会唤醒了,不在那个等待条件的队列里面了

// 那么就对当前线程再执行一次unpark,在下面 第 6 步  执行 park的时候就直接略过了,不会阻塞,也就不会出现问题

6.执行park操作把当前线程阻塞,调用的是 操作系统的wait 操作的 api

  6.1. 如果出现5的情况则当前线程并不会阻塞,而是直接走过去

7.如果线程醒来:有两种情况

  7.1. 其他线程执行了notify()操作,对应底层有unpark()操作,则该线程恢复运行

  7.2. 其他线程执行了interrupt()也会对当前线程有unpark()操作,该线程也会恢复运行

8.醒来好检查interrupt 标志位是否标示有异常,如果有,则会抛出 interruptedException 异常。

最后一条是线程在执行 obj.wait()有抛出interruptException异常的原因。

4,notify的原理

notify()操作将处于waitSet中的一个节点移动到

EntryList当中,也就是移动到对锁进行竞争而产生阻塞的队列当中。再等待其他锁的唤醒就ok了。

当然notify有一些唤醒策略。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,076评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,658评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,732评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,493评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,591评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,598评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,601评论 3 415
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,348评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,797评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,114评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,278评论 1 344
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,953评论 5 339
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,585评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,202评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,442评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,180评论 2 367
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,139评论 2 352

推荐阅读更多精彩内容