摘要:執行會重新將設置為,并且通知喚醒其中一個若有的話在方法中調用了函數而處于等待狀態的線程。除此之外,我們需要記錄同一個線程重復對一個鎖對象加鎖的次數。競爭失敗的線程處于就緒狀態,長期競爭失敗的線程就會饑餓。
tutorials site
Locks in javaLocks (and other more advanced synchronization mechanisms) are created using synchronized blocks, so it is not like we can get totally rid of the synchronized keyword.
鎖的實現是利用synchonized, wait(),notify()方法實現的。所以不可以認為鎖可以完全脫離synchonized實現。
Java包 JUC java.util.concurrent.locks 包括了很多lock接口的實現了類,這些類足夠使用。
但是需要知道如何使用它們,以及這些類背后的理論。JUC包教程
用synchonized:可以保證在同一時間只有一個線程可以執行 return ++count:
public class Counter{ private int count = 0; public int inc(){ synchronized(this){ return ++count; } } }
以下的Counter類用Lock代替synchronized 達到同樣的目的:
lock() 方法會對 Lock 實例對象進行加鎖,因此所有其他對該對象調用 lock() 方法的線程都會被阻塞,直到該 Lock 對象的 unlock() 方法被調用。
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類是怎么設計的?
Lock 類的設計一個Lock類的簡單實現:
javapublic 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) 循環, 又被稱為spin lock自旋鎖。當 isLocked 為 true 時,調用 lock() 的線程在 wait() 調用上阻塞等待。為防止該線程沒有收到 notify() 調用也從 wait() 中返回(也稱作虛假喚醒),這個線程會重新去檢查 isLocked 條件以決定當前是否可以安全地繼續執行還是需要重新保持等待,而不是認為線程被喚醒了就可以安全地繼續執行了。如果 isLocked 為 false,當前線程會退出 while(isLocked) 循環,并將 isLocked 設回 true,讓其它正在調用 lock() 方法的線程能夠在 Lock 實例上加鎖。
當線程完成了臨界區(位于 lock() 和 unlock() 之間)中的代碼,就會調用 unlock()。執行 unlock() 會重新將 isLocked 設置為 false,并且通知(喚醒)其中一個(若有的話)在 lock() 方法中調用了 wait() 函數而處于等待狀態的線程。
鎖的可重入性synchronized 同步塊是可重入的。這意味著: 如果一個java線程進入了代碼中的同步塊synchonzied block,并因此獲得了該同步塊使用的同步對象對應的管程monitor object上的鎖那么這個線程可以進入由同一個管程對象所同步的另一個 java 代碼塊
前面的Lock的設計就不是可重入的:
javapublic class Reentrant2{ Lock lock = new Lock(); public outer(){ lock.lock(); inner(); lock.unlock(); } public synchronized inner(){ lock.lock(); //do something lock.unlock(); } }
一個線程是否被允許退出 lock() 方法是由 while 循環(自旋鎖)中的條件決定的。當前的判斷條件是只有當 isLocked 為 false 時 lock 操作才被允許,而沒有考慮是哪個線程鎖住了它。
所以需要對Lock的設計做出如下修改,才能可重入。
javapublic 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(); } } } ... }
注意到現在的 while 循環(自旋鎖)也考慮到了已鎖住該 Lock 實例的線程。如果當前的鎖對象沒有被加鎖 (isLocked = false),或者當前調用線程已經對該 Lock 實例加了鎖,那么 while 循環就不會被執行,調用 lock() 的線程就可以退出該方法(譯者注:“被允許退出該方法” 在當前語義下就是指不會調用 wait() 而導致阻塞)。
除此之外,我們需要記錄同一個線程重復對一個鎖對象加鎖的次數。否則,一次 unblock() 調用就會解除整個鎖,即使當前鎖已經被加鎖過多次。在 unlock() 調用沒有達到對應 lock() 調用的次數之前,我們不希望鎖被解除。
現在這個 Lock 類就是可重入的了。
鎖的公平性Starvation and Fairness 饑餓和公平
一個線程因為其他線程長期占有CPU而自己獲得不到,這種狀態稱為Starvation. 解決線程饑餓的方法是公平機制fairness公平機制,讓所有線程都能公平的有機會去獲得CPU。
導致饑餓的原因高優先級的線程占有了所有CPU處理時間,這樣低優先級的線程獲得不到;
處于阻塞狀態的線程無限期被阻塞
Java 的同步代碼區也是一個導致饑餓的因素。Java 的同步代碼區對哪個線程允許進入的次序沒有任何保障。這就意味著理論上存在一個試圖進入該同步區的線程處于被永久堵塞的風險,因為其他線程總是能持續地先于它獲得訪問,這即是 “饑餓” 問題,而一個線程被 “饑餓致死” 正是因為它得不到 CPU 運行時間的機會
Java"s synchronized code blocks can be another cause of starvation.
處于等待狀態的對象無限期等待
如果多個線程處在 wait() 方法執行上,而對其調用 notify() 不會保證哪一個線程會獲得喚醒,任何線程都有可能處于繼續等待的狀態。因此存在這樣一個風險:一個等待線程從來得不到喚醒,因為其他等待線程總是能被獲得喚醒。
這里細說一下:多線程通過共享一個object對象,來調用對象的wait/notifyAll 來導致線程等待或者喚醒; 每次一個線程進入同步塊,其他所有線程陷入等待狀態;然后active線程調用notifyALL()函數喚醒所有等待線程,所有線程競爭,只有一個線程競爭成功,獲得CPU執行。競爭失敗的線程處于就緒狀態,長期競爭失敗的線程就會饑餓。
線程之間的對資源(object)競爭導致的饑餓,為了避免競爭,所以想辦法一次喚醒一個線程。也就是下面講的FairLock 公平鎖機制。
Implementing Fairness in Java使用鎖lock來代替同步塊synchonized block
每一個調用 lock() 的線程都會進入一個隊列,當解鎖后,只有隊列里的第一個線程 (隊首)被允許鎖住 Fairlock 實例,所有其它的線程都將處于等待狀態,直到他們處于隊列頭部。
公平鎖實現機制:為每一個線程創建一個專屬鎖對象(而非多個線程共享一個對象,來wait/notify()),然后用一個隊列來管理這些鎖對象,嘗試加鎖的線程會在各自的對象上等待,當一個線程unlock的時候,只通知隊列頭的鎖對象,以喚醒其對應的線程
為了讓這個 Lock 類具有可重入性,我們需要對它做一點小的改動:
javapublic class FairLock { private boolean isLocked = false; private Thread lockingThread = null; private ListwaitingThreads = new ArrayList (); public void lock() throws InterruptedException{ QueueObject queueObject = new QueueObject(); boolean isLockedForThisThread = true; synchronized(this){ waitingThreads.add(queueObject); } while(isLockedForThisThread){ synchronized(this){ isLockedForThisThread = isLocked || waitingThreads.get(0) != queueObject; if(!isLockedForThisThread){ isLocked = true; waitingThreads.remove(queueObject); lockingThread = Thread.currentThread(); return; } } try{ queueObject.doWait(); }catch(InterruptedException e){ synchronized(this) { waitingThreads.remove(queueObject); } throw e; } } } public synchronized void unlock(){ if(this.lockingThread != Thread.currentThread()){ throw new IllegalMonitorStateException( "Calling thread has not locked this lock"); } isLocked = false; lockingThread = null; if(waitingThreads.size() > 0){ waitingThreads.get(0).doNotify(); } } }
文章版權歸作者所有,未經允許請勿轉載,若此文章存在違規行為,您可以聯系管理員刪除。
轉載請注明本文地址:http://m.specialneedsforspecialkids.com/yun/64366.html
摘要:每個通過網絡到達服務器的連接都被包裝成一個任務并且傳遞給線程池。線程池的線程會并發的處理連接上的請求。用線程池控制線程數量,其他線程排隊等候。實現包,線程池頂級接口是但是嚴格意義講并不是一個線程。此線程池支持定時以及周期性執行任務的需求。 tutorial site1tutorial site2 一個問題: 每啟動一個新線程都會有相應的性能開銷(涉及到OS的交互:創建線程,銷毀線程...
摘要:請參看前一篇文章并發學習筆記一原子性可見性有序性問題六等待通知機制什么是等待通知機制當線程不滿足某個條件,則進入等待狀態如果線程滿足要求的某個條件后,則通知等待的線程重新執行。經極客時間并發編程實戰專欄內容學習整理 請參看前一篇文章:Java 并發學習筆記(一)——原子性、可見性、有序性問題 六、等待—通知機制 什么是等待通知—機制?當線程不滿足某個條件,則進入等待狀態;如果線程滿足要...
摘要:最后,總結一下,導致并發問題的三個源頭分別是原子性一個線程在執行的過程當中不被中斷。可見性一個線程修改了共享變量,另一個線程能夠馬上看到,就叫做可見性。 計算機的 CPU、內存、I/O 設備的速度一直存在較大的差異,依次是 CPU > 內存 > I/O 設備,為了權衡這三者的速度差異,主要提出了三種解決辦法: CPU 增加了緩存,均衡和內存的速度差異 發明了進程、線程,分時復用 CP...
閱讀 2682·2023-04-25 15:15
閱讀 1322·2021-11-25 09:43
閱讀 1611·2021-11-23 09:51
閱讀 1086·2021-11-12 10:36
閱讀 2888·2021-11-11 16:55
閱讀 962·2021-11-08 13:18
閱讀 734·2021-10-28 09:31
閱讀 2057·2019-08-30 15:47