欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

hotspot解析jdk1.8?Unsafe類(lèi)park和unpark方法使用

 更新時(shí)間:2023年01月09日 09:18:13   作者:linqiw  
這篇文章主要為大家介紹了hotspot解析jdk1.8?Unsafe類(lèi)park和unpark方法使用示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪

引言

Unsafe的park方法

park是Unsafe類(lèi)里的native方法,LockSupport類(lèi)通過(guò)調(diào)用Unsafe類(lèi)的park和unpark提供了幾個(gè)操作。Unsafe的park方法如下:

public native void park(boolean isAbsolute, long time);

 第一個(gè)參數(shù)是是否是絕對(duì)時(shí)間,第二個(gè)參數(shù)是等待時(shí)間值。如果isAbsolute是true則會(huì)實(shí)現(xiàn)ms定時(shí)。如果isAbsolute是false則會(huì)實(shí)現(xiàn)ns定時(shí)。

LockSupport類(lèi)常用的park方法如下,無(wú)參方法

public static void park() {
        UNSAFE.park(false, 0L);
}

執(zhí)行普通的掛起,isAbsolute是false,time是0。三種情況:1.在調(diào)用park()之前調(diào)用了unpark或者interrupt則park直接返回,不會(huì)掛起。2.如果未調(diào)用,則park會(huì)掛起當(dāng)前線程。3.park未知原因調(diào)用出錯(cuò)則直接返回(一般不會(huì)出現(xiàn))

實(shí)現(xiàn)ns計(jì)時(shí)的方法

 public static void parkNanos(long nanos) {
        if (nanos > 0)
            UNSAFE.park(false, nanos);
 }

isAbsolute是false,time大于0,則會(huì)實(shí)現(xiàn)高精度計(jì)時(shí)。三種情況:1.在調(diào)用park()之前調(diào)用了unpark或者interrupt則park直接返回,不會(huì)掛起。2.如果未調(diào)用則會(huì)掛起當(dāng)前線程,但是在掛起time ns時(shí)如果未收到喚醒信號(hào)也會(huì)返回繼續(xù)執(zhí)行。3.park未知原因調(diào)用出錯(cuò)則直接返回(一般不會(huì)出現(xiàn))

實(shí)現(xiàn)低精度的ms定時(shí)方法

 public static void parkUntil(long deadline) {
        UNSAFE.park(true, deadline);
 }

此時(shí)isAbsolute是true,time可以為任意數(shù)值。四種情況:

  • 1.在調(diào)用park()之前調(diào)用了unpark或者interrupt則park直接返回,不會(huì)掛起。
  • 2.如果time <= 0則直接返回。
  • 3.如果之前未調(diào)用park unpark并且time > 0,則會(huì)掛起當(dāng)前線程,但是在掛起time ms時(shí)如果未收到喚醒信號(hào)也會(huì)返回繼續(xù)執(zhí)行。
  • 4.park未知原因調(diào)用出錯(cuò)則直接返回(一般不會(huì)出現(xiàn))

hotspot對(duì)應(yīng)的類(lèi)

class Parker : public os::PlatformParker {
private:
  volatile int _counter ;   //計(jì)數(shù)
  Parker * FreeNext ;      //指向下一個(gè)Parker
  JavaThread * AssociatedWith ; // 指向parker所屬的線程。
public:
  Parker() : PlatformParker() {
    _counter       = 0 ;    //初始化為0
    FreeNext       = NULL ;
    AssociatedWith = NULL ;
  }
protected:
  ~Parker() { ShouldNotReachHere(); }
public:
  // For simplicity of interface with Java, all forms of park (indefinite,
  // relative, and absolute) are multiplexed into one call.
  void park(bool isAbsolute, jlong time);
  void unpark();
  // Lifecycle operators
  static Parker * Allocate (JavaThread * t) ;
  static void Release (Parker * e) ;private:
  static Parker * volatile FreeList ;
  static volatile int ListLock ;
};

Unsafe調(diào)用的park最終會(huì)調(diào)用Parker類(lèi)的park函數(shù),Parker繼承了PlatformParker。

class PlatformParker : public CHeapObj<mtInternal> {
  protected:
    enum {
        REL_INDEX = 0,
        ABS_INDEX = 1
    };
    int _cur_index;  // 條件變量數(shù)組下標(biāo),which cond is in use: -1, 0, 1
    pthread_mutex_t _mutex [1] ;  //pthread互斥鎖
    pthread_cond_t  _cond  [2] ; // pthread條件變量數(shù)組,一個(gè)用于相對(duì)時(shí)間,一個(gè)用于絕對(duì)時(shí)間。
  public:       // TODO-FIXME: make dtor private
    ~PlatformParker() { guarantee (0, "invariant") ; }
  public:
    PlatformParker() {
      int status;
      status = pthread_cond_init (&_cond[REL_INDEX], os::Linux::condAttr());
      assert_status(status == 0, status, "cond_init rel");
      status = pthread_cond_init (&_cond[ABS_INDEX], NULL);
      assert_status(status == 0, status, "cond_init abs");
      status = pthread_mutex_init (_mutex, NULL);
      assert_status(status == 0, status, "mutex_init");
      _cur_index = -1; // mark as unused
    }
};

PlatformParker主要看三個(gè)成員變量,_cur_index, _mutex, _cond。其中mutex和cond就是很熟悉的glibc nptl包中符合posix標(biāo)準(zhǔn)的線程同步工具,一個(gè)互斥鎖一個(gè)條件變量。再看thread和Parker的關(guān)系,在hotspot的Thread類(lèi)的NameThread內(nèi)部類(lèi)中有一個(gè) Parker成員變量。說(shuō)明parker是每線程變量,在創(chuàng)建線程的時(shí)候就會(huì)生成一個(gè)parker實(shí)例。

 // JSR166 per-thread parker
private:
  Parker*    _parker;

park的實(shí)現(xiàn)

void Parker::park(bool isAbsolute, jlong time) {
  //原子交換,如果_counter > 0,則將_counter置為0,直接返回,否則_counter為0
  if (Atomic::xchg(0, &_counter) > 0) return;
  //獲取當(dāng)前線程
  Thread* thread = Thread::current();
  assert(thread->is_Java_thread(), "Must be JavaThread");
  //下轉(zhuǎn)型為java線程
  JavaThread *jt = (JavaThread *)thread;
  //如果當(dāng)前線程設(shè)置了中斷標(biāo)志,調(diào)用park則直接返回,所以如果在park之前調(diào)用了
  //interrupt就會(huì)直接返回
  if (Thread::is_interrupted(thread, false)) {
    return;
  }
  // 高精度絕對(duì)時(shí)間變量
  timespec absTime;
  //如果time小于0,或者isAbsolute是true并且time等于0則直接返回
  if (time < 0 || (isAbsolute && time == 0) ) { // don't wait at all
    return;
  }
  //如果time大于0,則根據(jù)是否是高精度定時(shí)計(jì)算定時(shí)時(shí)間
  if (time > 0) {
    unpackTime(&absTime, isAbsolute, time);
  }
  //進(jìn)入安全點(diǎn)避免死鎖
  ThreadBlockInVM tbivm(jt);
  //如果當(dāng)前線程設(shè)置了中斷標(biāo)志,或者獲取mutex互斥鎖失敗則直接返回
  //由于Parker是每個(gè)線程都有的,所以_counter cond mutex都是每個(gè)線程都有的,
  //不是所有線程共享的所以加鎖失敗只有兩種情況,第一unpark已經(jīng)加鎖這時(shí)只需要返回即可,
  //第二調(diào)用調(diào)用pthread_mutex_trylock出錯(cuò)。對(duì)于第一種情況就類(lèi)似是unpark先調(diào)用的情況,所以
  //直接返回。
  if (Thread::is_interrupted(thread, false) || pthread_mutex_trylock(_mutex) != 0) {
    return;
  }
  int status ;
  //如果_counter大于0,說(shuō)明unpark已經(jīng)調(diào)用完成了將_counter置為了1,
  //現(xiàn)在只需將_counter置0,解鎖,返回
  if (_counter > 0)  { // no wait needed
    _counter = 0;
    status = pthread_mutex_unlock(_mutex);
    assert (status == 0, "invariant");
    OrderAccess::fence();
    return;
  }
  OSThreadWaitState osts(thread->osthread(), false /* not Object.wait() */);
  jt->set_suspend_equivalent();
  // cleared by handle_special_suspend_equivalent_condition() or java_suspend_self()
  assert(_cur_index == -1, "invariant");
  //如果time等于0,說(shuō)明是相對(duì)時(shí)間也就是isAbsolute是fasle(否則前面就直接返回了),則直接掛起
  if (time == 0) {
    _cur_index = REL_INDEX; // arbitrary choice when not timed
    status = pthread_cond_wait (&_cond[_cur_index], _mutex) ;
  } else { //如果time非0
    //判斷isAbsolute是false還是true,false的話使用_cond[0],否則用_cond[1]
    _cur_index = isAbsolute ? ABS_INDEX : REL_INDEX;
    //使用條件變量使得當(dāng)前線程掛起。
    status = os::Linux::safe_cond_timedwait (&_cond[_cur_index], _mutex, &absTime) ;
    //如果掛起失敗則銷(xiāo)毀當(dāng)前的條件變量重新初始化。
    if (status != 0 && WorkAroundNPTLTimedWaitHang) {
      pthread_cond_destroy (&_cond[_cur_index]) ;
      pthread_cond_init    (&_cond[_cur_index], isAbsolute ? NULL : os::Linux::condAttr());
    }
  }
  //如果pthread_cond_wait成功則以下代碼都是線程被喚醒后執(zhí)行的。
  _cur_index = -1;
  assert_status(status == 0 || status == EINTR ||
                status == ETIME || status == ETIMEDOUT,
                status, "cond_timedwait");
#ifdef ASSERT
  pthread_sigmask(SIG_SETMASK, &oldsigs, NULL);
#endif
  //將_counter變量重新置為1
  _counter = 0 ;
  //解鎖
  status = pthread_mutex_unlock(_mutex) ;
  assert_status(status == 0, status, "invariant") ;
  // 使用內(nèi)存屏障使_counter對(duì)其它線程可見(jiàn)
  OrderAccess::fence();
  // 如果在park線程掛起的時(shí)候調(diào)用了stop或者suspend則還需要將線程掛起不能返回
  if (jt->handle_special_suspend_equivalent_condition()) {
    jt->java_suspend_self();
  }
}

unpark函數(shù)

void Parker::unpark() {
  int s, status ;
  //加互斥鎖
  status = pthread_mutex_lock(_mutex);
  assert (status == 0, "invariant") ;
  s = _counter;
  _counter = 1; //將_counter置1
  //如果_counter是0則說(shuō)明調(diào)用了park或者沒(méi)調(diào)用(初始為counter0)
  //這也說(shuō)明park和unpark調(diào)用沒(méi)有先后順序。
  if (s < 1) {
    // 說(shuō)明當(dāng)前parker對(duì)應(yīng)的線程掛起了,因?yàn)開(kāi)cur_index初始是-1,并且等待條件變量的線程被喚醒
    //后也會(huì)將_cur_index重置-1
    if (_cur_index != -1) {
       //如果設(shè)置了WorkAroundNPTLTimedWaitHang先調(diào)用signal再調(diào)用unlock,否則相反
      //這兩個(gè)先后順序都可以,在hotspot在Linux下默認(rèn)使用這種方式
      //即先調(diào)用signal再調(diào)用unlock
      if (WorkAroundNPTLTimedWaitHang) {
        status = pthread_cond_signal (&_cond[_cur_index]);
        assert (status == 0, "invariant");
        status = pthread_mutex_unlock(_mutex);
        assert (status == 0, "invariant");
      } else {
        status = pthread_mutex_unlock(_mutex);
        assert (status == 0, "invariant");
        status = pthread_cond_signal (&_cond[_cur_index]);
        assert (status == 0, "invariant");
      }
    } else { //如果_cur_index == -1說(shuō)明線程沒(méi)在等待條件變量,則直接解鎖
      pthread_mutex_unlock(_mutex);
      assert (status == 0, "invariant") ;
    }
  } else {//如果_counter == 1,說(shuō)明線程調(diào)用了一次或多次unpark但是沒(méi)調(diào)用park,則直接解鎖
    pthread_mutex_unlock(_mutex);
    assert (status == 0, "invariant") ;
  }

unpark主要是根據(jù)counter和cur_index判斷當(dāng)前線程是否掛在條件變量上,如果是則signal,否則就什么也不做。

所以park和unpark和核心就是counter cur_index, mutex,cond,通過(guò)使用條件變量對(duì)counter進(jìn)行操作,在調(diào)用park的時(shí)候如果counter是0則會(huì)去執(zhí)行掛起的流程,否則返回,在掛起恢復(fù)后再將counter置為0。在unpark的時(shí)候如果counter是0則會(huì)執(zhí)行喚醒的流程,否則不執(zhí)行喚醒流程,并且不管什么情況始終將counter置為1。

以上就是hotspot解析jdk1.8 Unsafe類(lèi)park和unpark方法使用的詳細(xì)內(nèi)容,更多關(guān)于hotspot解析jdk Unsafe類(lèi)的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!

相關(guān)文章

  • SpringBoot整合SQLite數(shù)據(jù)庫(kù)全過(guò)程

    SpringBoot整合SQLite數(shù)據(jù)庫(kù)全過(guò)程

    sqlite是一個(gè)很輕量級(jí)的數(shù)據(jù)庫(kù),可以滿足日常sql的需求,下面這篇文章主要給大家介紹了關(guān)于SpringBoot整合SQLite數(shù)據(jù)庫(kù)的相關(guān)資料,文中通過(guò)實(shí)例代碼介紹的非常詳細(xì),需要的朋友可以參考下
    2023-03-03
  • 一文帶你了解gson如何序列化子類(lèi)

    一文帶你了解gson如何序列化子類(lèi)

    這篇文章主要為大家詳細(xì)介紹了gson如何序列化子類(lèi)的相關(guān)知識(shí),文中的示例代碼講解詳細(xì),具有一定的學(xué)習(xí)和借鑒價(jià)值,感興趣的小伙伴可以跟隨小編一起學(xué)習(xí)一下
    2023-10-10
  • Spring BeanName 的自動(dòng)生成原理示例詳解

    Spring BeanName 的自動(dòng)生成原理示例詳解

    這篇文章主要介紹了Spring BeanName 的自動(dòng)生成原理示例詳解,本文通過(guò)示例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2023-09-09
  • Java利用for循環(huán)輸出空心菱形的實(shí)例代碼

    Java利用for循環(huán)輸出空心菱形的實(shí)例代碼

    這篇文章主要介紹了Java利用for循環(huán)輸出空心菱形的實(shí)例代碼,需要的朋友可以參考下
    2014-02-02
  • Mybatis實(shí)現(xiàn)分包定義數(shù)據(jù)庫(kù)的原理與過(guò)程

    Mybatis實(shí)現(xiàn)分包定義數(shù)據(jù)庫(kù)的原理與過(guò)程

    這篇文章主要給大家介紹了關(guān)于Mybatis實(shí)現(xiàn)分包定義數(shù)據(jù)庫(kù)的原理與過(guò)程,文中通過(guò)實(shí)例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下
    2022-01-01
  • 教你在SpringBoot中管理多環(huán)境配置文件

    教你在SpringBoot中管理多環(huán)境配置文件

    在項(xiàng)目開(kāi)發(fā)中,一般會(huì)分為開(kāi)發(fā)環(huán)境,測(cè)試環(huán)境和生產(chǎn)環(huán)境.如果總是在一個(gè)文件中修修改改,難免會(huì)出現(xiàn)錯(cuò)誤,尤其是在交接的過(guò)程中,一旦出現(xiàn)問(wèn)題,輕則丟失績(jī)效,重則需要?jiǎng)h庫(kù)跑路.本文則帶大家詳細(xì)了解怎么管理多環(huán)境配置文件,需要的朋友可以參考下
    2021-05-05
  • Java中的可變參數(shù)常見(jiàn)用法實(shí)例總結(jié)

    Java中的可變參數(shù)常見(jiàn)用法實(shí)例總結(jié)

    這篇文章主要介紹了Java中的可變參數(shù)常見(jiàn)用法,結(jié)合實(shí)例形式總結(jié)分析了java可變參數(shù)的常見(jiàn)功能、使用方法及相關(guān)操作注意事項(xiàng),需要的朋友可以參考下
    2019-10-10
  • 半小時(shí)通透Java的泛型

    半小時(shí)通透Java的泛型

    這篇文章主要給大家介紹了關(guān)于Java中泛型使用的方法,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家學(xué)習(xí)或者使用Java具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面來(lái)一起學(xué)習(xí)學(xué)習(xí)吧
    2021-09-09
  • 在Idea2020.1中使用gitee2020.1.0創(chuàng)建第一個(gè)代碼庫(kù)的實(shí)現(xiàn)

    在Idea2020.1中使用gitee2020.1.0創(chuàng)建第一個(gè)代碼庫(kù)的實(shí)現(xiàn)

    這篇文章主要介紹了在Idea2020.1中使用gitee2020.1.0創(chuàng)建第一個(gè)代碼庫(kù)的實(shí)現(xiàn),文中通過(guò)圖文示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧
    2020-07-07
  • SpringCloud OpenFeign超詳細(xì)講解模板化遠(yuǎn)程通信的實(shí)現(xiàn)

    SpringCloud OpenFeign超詳細(xì)講解模板化遠(yuǎn)程通信的實(shí)現(xiàn)

    這篇文章主要介紹了SpringCloudSpringboot集成OpenFeign實(shí)現(xiàn)模板化遠(yuǎn)程通信,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧
    2022-07-07

最新評(píng)論