亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

java 多線程-鎖詳解及示例代碼

 更新時(shí)間:2016年09月02日 16:44:07   作者:DemonWang  
本文主要介紹 Java 多線程鎖的基礎(chǔ)知識(shí),這里整理了相關(guān)資料及示例代碼有興趣的小伙伴可以參考下

自 Java 5 開始,java.util.concurrent.locks 包中包含了一些鎖的實(shí)現(xiàn),因此你不用去實(shí)現(xiàn)自己的鎖了。但是你仍然需要去了解怎樣使用這些鎖。

一個(gè)簡(jiǎn)單的鎖

讓我們從 java 中的一個(gè)同步塊開始:

public class Counter{
  private int count = 0;

  public int inc(){
    synchronized(this){
      return ++count;
    }
  }
}

可以看到在 inc()方法中有一個(gè) synchronized(this)代碼塊。該代碼塊可以保證在同一時(shí)間只有一個(gè)線程可以執(zhí)行 return ++count。雖然在 synchronized 的同步塊中的代碼可以更加復(fù)雜,但是++count 這種簡(jiǎn)單的操作已經(jīng)足以表達(dá)出線程同步的意思。

以下的 Counter 類用 Lock 代替 synchronized 達(dá)到了同樣的目的:

public class Counter{
  private Lock lock = new Lock();
  private int count = 0;

  public int inc(){
    lock.lock();
    int newCount = ++count;
    lock.unlock();
    return newCount;
  }
}

lock()方法會(huì)對(duì) Lock 實(shí)例對(duì)象進(jìn)行加鎖,因此所有對(duì)該對(duì)象調(diào)用 lock()方法的線程都會(huì)被阻塞,直到該 Lock 對(duì)象的 unlock()方法被調(diào)用。

這里有一個(gè) Lock 類的簡(jiǎn)單實(shí)現(xiàn):

public class Counter{
public class Lock{
  private boolean isLocked = false;

  public synchronized void lock()
    throws InterruptedException{
    while(isLocked){
      wait();
    }
    isLocked = true;
  }

  public synchronized void unlock(){
    isLocked = false;
    notify();
  }
}

注意其中的 while(isLocked)循環(huán),它又被叫做“自旋鎖”。當(dāng) isLocked 為 true 時(shí),調(diào)用 lock()的線程在 wait()調(diào)用上阻塞等待。為防止該線程沒有收到 notify()調(diào)用也從 wait()中返回(也稱作虛假喚醒),這個(gè)線程會(huì)重新去檢查 isLocked 條件以決定當(dāng)前是否可以安全地繼續(xù)執(zhí)行還是需要重新保持等待,而不是認(rèn)為線程被喚醒了就可以安全地繼續(xù)執(zhí)行了。如果 isLocked 為 false,當(dāng)前線程會(huì)退出 while(isLocked)循環(huán),并將 isLocked 設(shè)回 true,讓其它正在調(diào)用 lock()方法的線程能夠在 Lock 實(shí)例上加鎖。

當(dāng)線程完成了臨界區(qū)(位于 lock()和 unlock()之間)中的代碼,就會(huì)調(diào)用 unlock()。執(zhí)行 unlock()會(huì)重新將 isLocked 設(shè)置為 false,并且通知(喚醒)其中一個(gè)(若有的話)在 lock()方法中調(diào)用了 wait()函數(shù)而處于等待狀態(tài)的線程。

鎖的可重入性

Java 中的 synchronized 同步塊是可重入的。這意味著如果一個(gè) java 線程進(jìn)入了代碼中的 synchronized 同步塊,并因此獲得了該同步塊使用的同步對(duì)象對(duì)應(yīng)的管程上的鎖,那么這個(gè)線程可以進(jìn)入由同一個(gè)管程對(duì)象所同步的另一個(gè) java 代碼塊。下面是一個(gè)例子:

public class Reentrant{
  public synchronized outer(){
    inner();
  }

  public synchronized inner(){
    //do something
  }
}

注意 outer()和 inner()都被聲明為 synchronized,這在 Java 中和 synchronized(this)塊等效。如果一個(gè)線程調(diào)用了 outer(),在 outer()里調(diào)用 inner()就沒有什么問(wèn)題,因?yàn)檫@兩個(gè)方法(代碼塊)都由同一個(gè)管程對(duì)象(”this”)所同步。如果一個(gè)線程已經(jīng)擁有了一個(gè)管程對(duì)象上的鎖,那么它就有權(quán)訪問(wèn)被這個(gè)管程對(duì)象同步的所有代碼塊。這就是可重入。線程可以進(jìn)入任何一個(gè)它已經(jīng)擁有的鎖所同步著的代碼塊。

前面給出的鎖實(shí)現(xiàn)不是可重入的。如果我們像下面這樣重寫 Reentrant 類,當(dāng)線程調(diào)用 outer()時(shí),會(huì)在 inner()方法的 lock.lock()處阻塞住。

public class Reentrant2{
  Lock lock = new Lock();

  public outer(){
    lock.lock();
    inner();
    lock.unlock();
  }

  public synchronized inner(){
    lock.lock();
    //do something
    lock.unlock();
  }
}

調(diào)用 outer()的線程首先會(huì)鎖住 Lock 實(shí)例,然后繼續(xù)調(diào)用 inner()。inner()方法中該線程將再一次嘗試鎖住 Lock 實(shí)例,結(jié)果該動(dòng)作會(huì)失?。ㄒ簿褪钦f(shuō)該線程會(huì)被阻塞),因?yàn)檫@個(gè) Lock 實(shí)例已經(jīng)在 outer()方法中被鎖住了。

兩次 lock()之間沒有調(diào)用 unlock(),第二次調(diào)用 lock 就會(huì)阻塞,看過(guò) lock()實(shí)現(xiàn)后,會(huì)發(fā)現(xiàn)原因很明顯:

public class Lock{
  boolean isLocked = false;

  public synchronized void lock()
    throws InterruptedException{
    while(isLocked){
      wait();
    }
    isLocked = true;
  }

  ...
}

一個(gè)線程是否被允許退出 lock()方法是由 while 循環(huán)(自旋鎖)中的條件決定的。當(dāng)前的判斷條件是只有當(dāng) isLocked 為 false 時(shí) lock 操作才被允許,而沒有考慮是哪個(gè)線程鎖住了它。

為了讓這個(gè) Lock 類具有可重入性,我們需要對(duì)它做一點(diǎn)小的改動(dòng):

public class Lock{
  boolean isLocked = false;
  Thread lockedBy = null;
  int lockedCount = 0;

  public synchronized void lock()
    throws InterruptedException{
    Thread callingThread =
      Thread.currentThread();
    while(isLocked && lockedBy != callingThread){
      wait();
    }
    isLocked = true;
    lockedCount++;
    lockedBy = callingThread;
 }

  public synchronized void unlock(){
    if(Thread.curentThread() ==
      this.lockedBy){
      lockedCount--;

      if(lockedCount == 0){
        isLocked = false;
        notify();
      }
    }
  }

  ...
}

注意到現(xiàn)在的 while 循環(huán)(自旋鎖)也考慮到了已鎖住該 Lock 實(shí)例的線程。如果當(dāng)前的鎖對(duì)象沒有被加鎖(isLocked = false),或者當(dāng)前調(diào)用線程已經(jīng)對(duì)該 Lock 實(shí)例加了鎖,那么 while 循環(huán)就不會(huì)被執(zhí)行,調(diào)用 lock()的線程就可以退出該方法(譯者注:“被允許退出該方法”在當(dāng)前語(yǔ)義下就是指不會(huì)調(diào)用 wait()而導(dǎo)致阻塞)。

除此之外,我們需要記錄同一個(gè)線程重復(fù)對(duì)一個(gè)鎖對(duì)象加鎖的次數(shù)。否則,一次 unblock()調(diào)用就會(huì)解除整個(gè)鎖,即使當(dāng)前鎖已經(jīng)被加鎖過(guò)多次。在 unlock()調(diào)用沒有達(dá)到對(duì)應(yīng) lock()調(diào)用的次數(shù)之前,我們不希望鎖被解除。

現(xiàn)在這個(gè) Lock 類就是可重入的了。

鎖的公平性

Java 的 synchronized 塊并不保證嘗試進(jìn)入它們的線程的順序。因此,如果多個(gè)線程不斷競(jìng)爭(zhēng)訪問(wèn)相同的 synchronized 同步塊,就存在一種風(fēng)險(xiǎn),其中一個(gè)或多個(gè)線程永遠(yuǎn)也得不到訪問(wèn)權(quán) —— 也就是說(shuō)訪問(wèn)權(quán)總是分配給了其它線程。這種情況被稱作線程饑餓。為了避免這種問(wèn)題,鎖需要實(shí)現(xiàn)公平性。本文所展現(xiàn)的鎖在內(nèi)部是用 synchronized 同步塊實(shí)現(xiàn)的,因此它們也不保證公平性。

在 finally 語(yǔ)句中調(diào)用 unlock()

如果用 Lock 來(lái)保護(hù)臨界區(qū),并且臨界區(qū)有可能會(huì)拋出異常,那么在 finally 語(yǔ)句中調(diào)用 unlock()就顯得非常重要了。這樣可以保證這個(gè)鎖對(duì)象可以被解鎖以便其它線程能繼續(xù)對(duì)其加鎖。以下是一個(gè)示例:

lock.lock();
try{
  //do critical section code,
  //which may throw exception
} finally {
  lock.unlock();
}

這個(gè)簡(jiǎn)單的結(jié)構(gòu)可以保證當(dāng)臨界區(qū)拋出異常時(shí) Lock 對(duì)象可以被解鎖。如果不是在 finally 語(yǔ)句中調(diào)用的 unlock(),當(dāng)臨界區(qū)拋出異常時(shí),Lock 對(duì)象將永遠(yuǎn)停留在被鎖住的狀態(tài),這會(huì)導(dǎo)致其它所有在該 Lock 對(duì)象上調(diào)用 lock()的線程一直阻塞。

以上就是關(guān)于 java 多線程鎖的資料整理,后續(xù)繼續(xù)補(bǔ)充相關(guān)資料,謝謝大家對(duì)本站的支持!

相關(guān)文章

  • Netty分布式客戶端處理接入事件handle源碼解析

    Netty分布式客戶端處理接入事件handle源碼解析

    這篇文章主要為大家介紹了Netty源碼分析客戶端流程處理接入事件handle創(chuàng)建,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2022-03-03
  • springboot?@PostConstruct無(wú)效的解決

    springboot?@PostConstruct無(wú)效的解決

    這篇文章主要介紹了springboot?@PostConstruct無(wú)效的解決,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2021-11-11
  • Activiti流程圖查看實(shí)例

    Activiti流程圖查看實(shí)例

    這篇文章主要介紹了Activiti流程圖查看實(shí)例,需要的朋友可以參考下
    2014-08-08
  • Java并發(fā) 線程間的等待與通知

    Java并發(fā) 線程間的等待與通知

    這篇文章主要介紹了Java并發(fā) 線程間的等待與通知,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下
    2019-10-10
  • Java經(jīng)典設(shè)計(jì)模式之適配器模式原理與用法詳解

    Java經(jīng)典設(shè)計(jì)模式之適配器模式原理與用法詳解

    這篇文章主要介紹了Java經(jīng)典設(shè)計(jì)模式之適配器模式,簡(jiǎn)單說(shuō)明了適配器模式的概念、原理,并結(jié)合實(shí)例形式分析了java適配器模式的用法與相關(guān)注意事項(xiàng),需要的朋友可以參考下
    2017-08-08
  • SpringBoot集成swagger的實(shí)例代碼

    SpringBoot集成swagger的實(shí)例代碼

    Swagger 是一款RESTFUL接口的文檔在線自動(dòng)生成+功能測(cè)試功能軟件,這篇文章主要介紹了SpringBoot集成swagger,需要的朋友可以參考下
    2017-12-12
  • Java ArrayDeque使用方法詳解

    Java ArrayDeque使用方法詳解

    這篇文章主要為大家詳細(xì)介紹了Java ArrayDeque的使用方法,感興趣的小伙伴們可以參考一下
    2016-03-03
  • Java中super關(guān)鍵字的用法和細(xì)節(jié)

    Java中super關(guān)鍵字的用法和細(xì)節(jié)

    大家好,本篇文章主要講的是Java中super關(guān)鍵字的用法和細(xì)節(jié),感興趣的同學(xué)趕快來(lái)看一看吧,對(duì)你有幫助的話記得收藏一下
    2022-01-01
  • 關(guān)于Java中常見的負(fù)載均衡算法

    關(guān)于Java中常見的負(fù)載均衡算法

    這篇文章主要介紹了關(guān)于Java中常見的負(fù)載均衡算法,負(fù)載平衡是一種電子計(jì)算機(jī)技術(shù),用來(lái)在多個(gè)計(jì)算機(jī)、網(wǎng)絡(luò)連接、CPU、磁盤驅(qū)動(dòng)器或其他資源中分配負(fù)載,以達(dá)到優(yōu)化資源使用、最大化吞吐率、最小化響應(yīng)時(shí)間、同時(shí)避免過(guò)載的目的,需要的朋友可以參考下
    2023-08-08
  • 利用Socket.io 實(shí)現(xiàn)消息實(shí)時(shí)推送功能

    利用Socket.io 實(shí)現(xiàn)消息實(shí)時(shí)推送功能

    這篇文章主要介紹了利用Socket.io 實(shí)現(xiàn)消息實(shí)時(shí)推送功能,需要的朋友可以參考下
    2017-12-12

最新評(píng)論