Java中的ReentrantLock和synchronized兩種鎖定機制的對比

多線程和併發性並非什麼新內容,可是 Java 語言設計中的創新之一就是,它是第一個直接把跨平臺線程模型和正規的內存模型集成到語言中的主流語言。核心類庫包含一個 Thread 類,能夠用它來構建、啓動和操縱線程,Java 語言包括了跨線程傳達併發性約束的構造 —— synchronized 和 volatile 。在簡化與平臺無關的併發類的開發的同時,它決沒有使併發類的編寫工做變得更繁瑣,只是使它變得更容易了。java

synchronized 快速回顧算法

把代碼塊聲明爲 synchronized,有兩個重要後果,一般是指該代碼具備 原子性(atomicity)和 可見性(visibility)。原子性意味着一個線程一次只能執行由一個指定監控對象(lock)保護的代碼,從而防止多個線程在更新共享狀態時相互衝突。可見性則更爲微妙;它要對付內存緩存和編譯器優化的各類反常行爲。通常來講,線程以某種沒必要讓其餘線程當即能夠看到的方式(無論這些線程在寄存器中、在處理器特定的緩存中,仍是經過指令重排或者其餘編譯器優化),不受緩存變量值的約束,可是若是開發人員使用了同步,以下面的代碼所示,那麼運行庫將確保某一線程對變量所作的更新先於對現有 synchronized 塊所進行的更新,當進入由同一監控器(lock)保護的另外一個 synchronized 塊時,將馬上能夠看到這些對變量所作的更新。相似的規則也存在於 volatile 變量上。編程

synchronized (lockObject) { 
  // update object state
}

因此,實現同步操做須要考慮安全更新多個共享變量所需的一切,不能有爭用條件,不能破壞數據(假設同步的邊界位置正確),並且要保證正確同步的其餘線程能夠看到這些變量的最新值。經過定義一個清晰的、跨平臺的內存模型(該模型在 JDK 5.0 中作了修改,改正了原來定義中的某些錯誤),經過遵照下面這個簡單規則,構建「一次編寫,隨處運行」的併發類是有可能的:緩存

不論何時,只要您將編寫的變量接下來可能被另外一個線程讀取,或者您將讀取的變量最後是被另外一個線程寫入的,那麼您必須進行同步。

不過如今好了一點,在最近的 JVM 中,沒有爭用的同步(一個線程擁有鎖的時候,沒有其餘線程企圖得到鎖)的性能成本仍是很低的。(也不老是這樣;早期 JVM 中的同步尚未優化,因此讓不少人都這樣認爲,可是如今這變成了一種誤解,人們認爲無論是否是爭用,同步都有很高的性能成本。)安全


對 synchronized 的改進多線程

如此看來同步至關好了,是麼?那麼爲何 JSR 166 小組花了這麼多時間來開發 java.util.concurrent.lock 框架呢?答案很簡單-同步是不錯,但它並不完美。它有一些功能性的限制 —— 它沒法中斷一個正在等候得到鎖的線程,也沒法經過投票獲得鎖,若是不想等下去,也就無法獲得鎖。同步還要求鎖的釋放只能在與得到鎖所在的堆棧幀相同的堆棧幀中進行,多數狀況下,這沒問題(並且與異常處理交互得很好),可是,確實存在一些非塊結構的鎖定更合適的狀況。併發

ReentrantLock 類框架

java.util.concurrent.lock 中的 Lock 框架是鎖定的一個抽象,它容許把鎖定的實現做爲 Java 類,而不是做爲語言的特性來實現。這就爲Lock 的多種實現留下了空間,各類實現可能有不一樣的調度算法、性能特性或者鎖定語義。 ReentrantLock 類實現了 Lock ,它擁有與 synchronized 相同的併發性和內存語義,可是添加了相似鎖投票、定時鎖等候和可中斷鎖等候的一些特性。此外,它還提供了在激烈爭用狀況下更佳的性能。(換句話說,當許多線程都想訪問共享資源時,JVM 能夠花更少的時候來調度線程,把更多時間用在執行線程上。)dom

reentrant 鎖意味着什麼呢?簡單來講,它有一個與鎖相關的獲取計數器,若是擁有鎖的某個線程再次獲得鎖,那麼獲取計數器就加1,而後鎖須要被釋放兩次才能得到真正釋放。這模仿了 synchronized 的語義;若是線程進入由線程已經擁有的監控器保護的 synchronized 塊,就容許線程繼續進行,當線程退出第二個(或者後續) synchronized 塊的時候,不釋放鎖,只有線程退出它進入的監控器保護的第一個 synchronized 塊時,才釋放鎖。工具

在查看清單 1 中的代碼示例時,能夠看到 Lock 和 synchronized 有一點明顯的區別 —— lock 必須在 finally 塊中釋放。不然,若是受保護的代碼將拋出異常,鎖就有可能永遠得不到釋放!這一點區別看起來可能沒什麼,可是實際上,它極爲重要。忘記在 finally 塊中釋放鎖,可能會在程序中留下一個定時炸彈,當有一天炸彈爆炸時,您要花費很大力氣纔有找到源頭在哪。而使用同步,JVM 將確保鎖會得到自動釋放。


清單 1. 用 ReentrantLock 保護代碼塊。

Lock lock = new ReentrantLock();
lock.lock();
try { 
  // update object state
}
finally {
  lock.unlock(); 
}


除此以外,與目前的 synchronized 實現相比,爭用下的 ReentrantLock 實現更具可伸縮性。(在將來的 JVM 版本中,synchronized 的爭用性能頗有可能會得到提升。)這意味着當許多線程都在爭用同一個鎖時,使用 ReentrantLock 的整體開支一般要比 synchronized少得多。


比較 ReentrantLock 和 synchronized 的可伸縮性

Tim Peierls 用一個簡單的線性全等僞隨機數生成器(PRNG)構建了一個簡單的評測,用它來測量 synchronized 和 Lock 之間相對的可伸縮性。這個示例很好,由於每次調用 nextRandom() 時,PRNG 都確實在作一些工做,因此這個基準程序其實是在測量一個合理的、真實的 synchronized 和 Lock 應用程序,而不是測試純粹紙上談兵或者什麼也不作的代碼(就像許多所謂的基準程序同樣。)

在這個基準程序中,有一個 PseudoRandom 的接口,它只有一個方法 nextRandom(int bound) 。該接口與 java.util.Random 類的功能很是相似。由於在生成下一個隨機數時,PRNG 用最新生成的數字做爲輸入,並且把最後生成的數字做爲一個實例變量來維護,其重點在於讓更新這個狀態的代碼段不被其餘線程搶佔,因此我要用某種形式的鎖定來確保這一點。( java.util.Random 類也能夠作到這點。)咱們爲 PseudoRandom 構建了兩個實現;一個使用 syncronized,另外一個使用 java.util.concurrent.ReentrantLock 。驅動程序生成了大量線程,每一個線程都瘋狂地爭奪時間片,而後計算不一樣版本每秒能執行多少輪。圖 1 和 圖 2 總結了不一樣線程數量的結果。這個評測並不完美,並且只在兩個系統上運行了(一個是雙 Xeon 運行超線程 Linux,另外一個是單處理器 Windows 系統),可是,應當足以表現 synchronized 與 ReentrantLock 相比所具備的伸縮性優點了。

圖 1 和圖 2 中的圖表以每秒調用數爲單位顯示了吞吐率,把不一樣的實現調整到 1 線程 synchronized 的狀況。每一個實現都相對迅速地集中在某個穩定狀態的吞吐率上,該狀態一般要求處理器獲得充分利用,把大多數的處理器時間都花在處理實際工做(計算機隨機數)上,只有小部分時間花在了線程調度開支上。您會注意到,synchronized 版本在處理任何類型的爭用時,表現都至關差,而 Lock 版本在調度的開支上花的時間至關少,從而爲更高的吞吐率留下空間,實現了更有效的 CPU 利用。


條件變量

根類 Object 包含某些特殊的方法,用來在線程的 wait() 、 notify() 和 notifyAll() 之間進行通訊。這些是高級的併發性特性,許多開發人員歷來沒有用過它們 —— 這多是件好事,由於它們至關微妙,很容易使用不當。幸運的是,隨着 JDK 5.0 中引入java.util.concurrent ,開發人員幾乎更加沒有什麼地方須要使用這些方法了。

通知與鎖定之間有一個交互 —— 爲了在對象上 wait 或 notify ,您必須持有該對象的鎖。就像 Lock 是同步的歸納同樣, Lock 框架包含了對 wait 和 notify 的歸納,這個歸納叫做 條件(Condition) 。 Lock 對象則充當綁定到這個鎖的條件變量的工廠對象,與標準的 wait 和 notify 方法不一樣,對於指定的 Lock ,能夠有不止一個條件變量與它關聯。這樣就簡化了許多併發算法的開發。例如, 條件(Condition) 的 Javadoc 顯示了一個有界緩衝區實現的示例,該示例使用了兩個條件變量,「not full」和「not empty」,它比每一個 lock 只用一個 wait 設置的實現方式可讀性要好一些(並且更有效)。 Condition 的方法與 wait 、 notify 和 notifyAll 方法相似,分別命名爲 await 、 signal 和 signalAll ,由於它們不能覆蓋 Object 上的對應方法。


這不公平

若是查看 Javadoc,您會看到, ReentrantLock 構造器的一個參數是 boolean 值,它容許您選擇想要一個 公平(fair)鎖,仍是一個 不公平(unfair)鎖。公平鎖使線程按照請求鎖的順序依次得到鎖;而不公平鎖則容許討價還價,在這種狀況下,線程有時能夠比先請求鎖的其餘線程先獲得鎖。

爲何咱們不讓全部的鎖都公平呢?畢竟,公平是好事,不公平是很差的,不是嗎?(當孩子們想要一個決定時,總會叫嚷「這不公平」。咱們認爲公平很是重要,孩子們也知道。)在現實中,公平保證了鎖是很是健壯的鎖,有很大的性能成本。要確保公平所須要的記賬(bookkeeping)和同步,就意味着被爭奪的公平鎖要比不公平鎖的吞吐率更低。做爲默認設置,應當把公平設置爲 false ,除非公平對您的算法相當重要,須要嚴格按照線程排隊的順序對其進行服務。

那麼同步又如何呢?內置的監控器鎖是公平的嗎?答案令許多人感到大吃一驚,它們是不公平的,並且永遠都是不公平的。可是沒有人抱怨過線程飢渴,由於 JVM 保證了全部線程最終都會獲得它們所等候的鎖。確保統計上的公平性,對多數狀況來講,這就已經足夠了,而這花費的成本則要比絕對的公平保證的低得多。因此,默認狀況下 ReentrantLock 是「不公平」的,這一事實只是把同步中一直是事件的東西表面化而已。若是您在同步的時候並不介意這一點,那麼在 ReentrantLock 時也沒必要爲它擔憂。

圖 3 和圖 4 包含與 圖 1和 圖 2 相同的數據,只是添加了一個數據集,用來進行隨機數基準檢測,此次檢測使用了公平鎖,而不是默認的協商鎖。正如您能看到的,公平是有代價的。若是您須要公平,就必須付出代價,可是請不要把它做爲您的默認選擇。


到處都好?

看起來 ReentrantLock 不管在哪方面都比 synchronized 好 —— 全部 synchronized 能作的,它都能作,它擁有與 synchronized 相同的內存和併發性語義,還擁有 synchronized 所沒有的特性,在負荷下還擁有更好的性能。那麼,咱們是否是應當忘記 synchronized ,再也不把它看成已經已經獲得優化的好主意呢?或者甚至用 ReentrantLock 重寫咱們現有的 synchronized 代碼?實際上,幾本 Java 編程方面介紹性的書籍在它們多線程的章節中就採用了這種方法,徹底用 Lock 來作示例,只把 synchronized 看成歷史。但我以爲這是把好事作得太過了。

還不要拋棄 synchronized

雖然 ReentrantLock 是個很是動人的實現,相對 synchronized 來講,它有一些重要的優點,可是我認爲急於把 synchronized 視若敝屣,絕對是個嚴重的錯誤。 java.util.concurrent.lock 中的鎖定類是用於高級用戶和高級狀況的工具 。通常來講,除非您對 Lock 的某個高級特性有明確的須要,或者有明確的證據(而不是僅僅是懷疑)代表在特定狀況下,同步已經成爲可伸縮性的瓶頸,不然仍是應當繼續使用 synchronized。

爲何我在一個顯然「更好的」實現的使用上主張保守呢?由於對於 java.util.concurrent.lock 中的鎖定類來講,synchronized 仍然有一些優點。好比,在使用 synchronized 的時候,不能忘記釋放鎖;在退出 synchronized 塊時,JVM 會爲您作這件事。您很容易忘記用 finally 塊釋放鎖,這對程序很是有害。您的程序可以經過測試,但會在實際工做中出現死鎖,那時會很難指出緣由(這也是爲何根本不讓初級開發人員使用 Lock 的一個好理由。)

另外一個緣由是由於,當 JVM 用 synchronized 管理鎖定請求和釋放時,JVM 在生成線程轉儲時可以包括鎖定信息。這些對調試很是有價值,由於它們能標識死鎖或者其餘異常行爲的來源。 Lock 類只是普通的類,JVM 不知道具體哪一個線程擁有 Lock 對象。並且,幾乎每一個開發人員都熟悉 synchronized,它能夠在 JVM 的全部版本中工做。在 JDK 5.0 成爲標準(從如今開始可能須要兩年)以前,使用 Lock類將意味着要利用的特性不是每一個 JVM 都有的,並且不是每一個開發人員都熟悉的。

何時選擇用 ReentrantLock 代替 synchronized

既然如此,咱們何時才應該使用 ReentrantLock 呢?答案很是簡單 —— 在確實須要一些 synchronized 所沒有的特性的時候,好比時間鎖等候、可中斷鎖等候、無塊結構鎖、多個條件變量或者鎖投票。 ReentrantLock 還具備可伸縮性的好處,應當在高度爭用的狀況下使用它,可是請記住,大多數 synchronized 塊幾乎歷來沒有出現過爭用,因此能夠把高度爭用放在一邊。我建議用 synchronized 開發,直到確實證實 synchronized 不合適,而不要僅僅是假設若是使用 ReentrantLock 「性能會更好」。請記住,這些是供高級用戶使用的高級工具。(並且,真正的高級用戶喜歡選擇可以找到的最簡單工具,直到他們認爲簡單的工具不適用爲止。)。一如既往,首先要把事情作好,而後再考慮是否是有必要作得更快

相關文章
相關標籤/搜索