當(dāng)前位置:首頁 > 技術(shù)學(xué)院 > 技術(shù)前線
[導(dǎo)讀]在并發(fā)編程中,鎖是保護(hù)共享資源的重要機(jī)制。然而,不正確的鎖使用可能會導(dǎo)致性能下降、死鎖等問題。因此,對鎖進(jìn)行調(diào)優(yōu)是提高并發(fā)程序性能和穩(wěn)定性的關(guān)鍵之一。

在并發(fā)編程中,鎖是保護(hù)共享資源的重要機(jī)制。然而,不正確的鎖使用可能會導(dǎo)致性能下降、死鎖等問題。因此,對鎖進(jìn)行調(diào)優(yōu)是提高并發(fā)程序性能和穩(wěn)定性的關(guān)鍵之一。本文將介紹一些常用的鎖調(diào)優(yōu)技巧,幫助您更好地優(yōu)化并發(fā)程序性能。

1. 并發(fā)編程和鎖的概念

并發(fā)編程,簡而言之,就是同時運行多個任務(wù)。在一個具有多個處理器的系統(tǒng)中,這意味著可以同時執(zhí)行多個任務(wù)。而在只有一個處理器的系統(tǒng)中,雖然一次只能執(zhí)行一個任務(wù),但由于任務(wù)之間的切換速度非??欤o我們的感覺就像所有任務(wù)都在同時運行。在 Java 中,我們通常使用線程來實現(xiàn)并發(fā)編程。了解更多并發(fā)編程的基礎(chǔ)知識,可以訪問這里。

1.1. 鎖的基本概念

在并發(fā)編程中,我們常常會遇到多個線程同時訪問和修改同一份數(shù)據(jù)的情況。為了保證數(shù)據(jù)的一致性和正確性,我們需要使用到鎖的概念。鎖可以防止多個線程同時修改同一份數(shù)據(jù),保證在任何時刻,只有一個線程能修改數(shù)據(jù)。

在 Java 中,我們主要使用兩種類型的鎖:互斥鎖和讀寫鎖?;コ怄i保證同一時刻只有一個線程能訪問某一共享資源。在 Java 中,我們可以通過 synchronized 關(guān)鍵字來實現(xiàn)互斥鎖。另一種鎖是讀寫鎖,它允許多個線程同時讀取共享資源,但在寫入數(shù)據(jù)時,只能有一個線程進(jìn)行,其他所有線程(無論是讀線程還是寫線程)都無法訪問共享資源。在 Java 中,我們可以使用 ReentrantReadWriteLock 類來實現(xiàn)讀寫鎖。

讓我們來看一個簡單的互斥鎖的示例。在下面的代碼中,我們創(chuàng)建了一個對象 lock,并使用 synchronized 關(guān)鍵字對這個對象進(jìn)行加鎖。這樣,在執(zhí)行 criticalSection() 方法的過程中,只有獲得 lock 對象的鎖的線程才能執(zhí)行。

Object lock = new Object();

synchronized(lock) {

criticalSection();

}

對于讀寫鎖,我們可以使用 ReentrantReadWriteLock 類來實現(xiàn)。以下是一個簡單的示例:

ReentrantReadWriteLock lock = new ReentrantReadWriteLock();

// 獲取讀鎖

lock.readLock().lock();

try {

readData();

} finally {

lock.readLock().unlock();

}

// 獲取寫鎖

lock.writeLock().lock();

try {

writeData();

} finally {

lock.writeLock().unlock();

}

2. 鎖優(yōu)化

2.1. 概念和原理

什么是鎖優(yōu)化

鎖優(yōu)化,簡單來說,就是通過一些技術(shù)手段來改進(jìn)鎖的使用方式,以提高并發(fā)程序的運行效率。這些技術(shù)手段包括但不限于:鎖粗化,鎖消除,輕量級鎖,偏向鎖等。

鎖優(yōu)化的方式

鎖粗化:這是一種將多次連續(xù)的鎖定操作合并為一次的優(yōu)化手段。假如一個線程在一段代碼中反復(fù)對同一個對象進(jìn)行加鎖和解鎖,那么 JVM 就會將這些鎖的范圍擴(kuò)大(粗化),即在第一次加鎖的位置加鎖,最后一次解鎖的位置解鎖,中間的加鎖解鎖操作則被省略。鎖消除:這是一種刪除不必要的鎖操作的優(yōu)化手段。在 Java 程序中,有些鎖實際上是不必要的,例如在只會被一個線程使用的數(shù)據(jù)上加的鎖。JVM 在 JIT 編譯的時候,通過一種叫做逃逸分析的技術(shù),可以檢測到這些不必要的鎖,然后將其刪除。輕量級鎖:這是一種在無競爭情況下,減少不必要的重量級鎖性能消耗的優(yōu)化手段。如果在獲取鎖的時候沒有競爭,那么 JVM 就會使用輕量級鎖。如果后續(xù)有競爭出現(xiàn),輕量級鎖就會膨脹為重量級鎖。偏向鎖:這是一種針對只有一個線程訪問同步代碼塊的情況的優(yōu)化手段。如果一個鎖主要被一個線程所獲取,那么 JVM 就會讓這個線程"偏向"這個鎖,后續(xù)這個線程再獲取這個鎖,就無需進(jìn)行額外的同步操作。這大大提高了鎖的獲取速度。3. 鎖消除

何時可以進(jìn)行鎖消除

鎖消除主要應(yīng)用在沒有多線程競爭的情況下。具體來說,當(dāng)一個數(shù)據(jù)僅在一個線程中使用,或者說這個數(shù)據(jù)的作用域僅限于一個線程時,這個線程對該數(shù)據(jù)的所有操作都不需要加鎖。在 Java HotSpot VM 中,這種優(yōu)化主要是通過逃逸分析(Escape Analysis)來實現(xiàn)的。

為什么鎖消除有效

鎖消除之所以有效,是因為它消除了不必要的鎖競爭,從而減少了線程切換和線程調(diào)度帶來的性能開銷。當(dāng)數(shù)據(jù)僅在單個線程中使用時,對此數(shù)據(jù)的所有操作都不需要同步。在這種情況下,鎖操作不僅不會增加安全性,反而會因為增加了額外的執(zhí)行開銷而降低程序的運行效率。

如何在代碼中實現(xiàn)鎖消除

在代碼層面上,我們無法直接控制 JVM 進(jìn)行鎖消除優(yōu)化,這是由 JVM 的 JIT 編譯器在運行時動態(tài)完成的。但我們可以通過編寫高質(zhì)量的代碼,使 JIT 編譯器更容易識別出可以進(jìn)行鎖消除的場景。例如:

public class LockElimination {

public void appendString(String str1, String str2, String str3) {

StringBuffer sb = new StringBuffer();

sb.append(str1).append(str2).append(str3);

System.out.println(sb.toString());

}

}

在這段代碼中,StringBuffer 實例 sb 的作用域僅限于 appendString 方法。在多線程環(huán)境中,不同的線程執(zhí)行 appendString 方法會創(chuàng)建各自的 StringBuffer 實例,互不影響。因此,JIT 編譯器會發(fā)現(xiàn)這種情況并自動消除 sb.append 操作中的鎖競爭。

所以,就有了如下編碼規(guī)則:

變量作用域應(yīng)盡可能小

鎖消除是一種有效的優(yōu)化手段,它可以幫助我們消除不必要的鎖,從而提高程序的運行效率。在日常編程中,我們應(yīng)該盡量避免在單線程的上下文中使用同步數(shù)據(jù)結(jié)構(gòu),從而使得鎖消除技術(shù)得以發(fā)揮作用。

4. 鎖粗化

何時可以進(jìn)行鎖粗化

鎖粗化,簡單來說,就是將多個連續(xù)的鎖擴(kuò)展為一個更大范圍的鎖。也就是說,如果 JVM 檢測到有連續(xù)的對同一對象的加鎖、解鎖操作,就會把這些加鎖、解鎖操作合并為對這段區(qū)域進(jìn)行一次連續(xù)的加鎖和解鎖。具體的示例如下:

synchronized (lock) {

// 代碼塊 1

}

// 無關(guān)代碼

synchronized (lock) {

// 代碼塊 2

}

JVM 在運行時可能會選擇將上述兩個小的同步塊合并,形成一個大的同步塊:

synchronized (lock) {

// 代碼塊 1

// 無關(guān)代碼

// 代碼塊 2

}

為什么鎖粗化有效

加鎖和解鎖操作本身也會帶來一定的性能開銷,因為每次加鎖和解鎖都可能會涉及到線程切換、線程調(diào)度等開銷。如果有大量小的同步塊頻繁地進(jìn)行加鎖和解鎖,那么這部分開銷可能會變得很大,從而降低程序的執(zhí)行效率。

通過鎖粗化,可以將多次加鎖和解鎖操作減少到一次,從而減少這部分開銷,提高程序的運行效率。

如何在代碼中實現(xiàn)鎖粗化

在代碼層面上,我們并不能直接控制 JVM 進(jìn)行鎖粗化,因為這是 JVM 在運行時動態(tài)進(jìn)行的優(yōu)化。不過,我們可以在編寫代碼時,盡量減少不必要的同步塊,避免頻繁加鎖和解鎖。這樣,就為 JVM 的鎖粗化優(yōu)化提供了可能。

鎖粗化是 JVM 提供的一種優(yōu)化手段,能夠有效地提高并發(fā)編程的效率。在我們編寫并發(fā)代碼時,應(yīng)當(dāng)注意同步塊的使用,盡量減少不必要的加鎖和解鎖,從而使得鎖粗化技術(shù)能夠發(fā)揮作用。

5. 鎖優(yōu)化與鎖粗化的選擇

鎖優(yōu)化使用場景

大量重入的場景:例如,當(dāng)一個方法大量地調(diào)用自身或者其他同步方法時,每次調(diào)用都需要加鎖、解鎖,這在極端情況下可能導(dǎo)致系統(tǒng)開銷大增。此時可以考慮使用鎖優(yōu)化。頻繁請求同一個鎖的場景:當(dāng)多個線程頻繁地請求同一個鎖時,鎖優(yōu)化可以減少鎖請求次數(shù),從而提高性能。鎖粗化使用場景

短時間內(nèi)多次獲取和釋放同一把鎖的場景:如果在短時間內(nèi),一段代碼多次獲取和釋放同一把鎖,這種情況下可以考慮使用鎖粗化,將多個連續(xù)的鎖合并為一個更大的鎖。沒有競爭的場景:如果在沒有競爭的情況下,仍然存在大量的加鎖、解鎖操作,這將導(dǎo)致不必要的性能損耗。在這種情況下,鎖粗化可以有效地減少加鎖、解鎖的次數(shù),從而提高性能。鎖優(yōu)化和鎖粗化都是為了提高程序的并發(fā)性能。具體應(yīng)用哪種方法,需要根據(jù)實際的代碼和運行情況進(jìn)行選擇。

6. Java 中的鎖優(yōu)化和鎖粗化

鎖優(yōu)化

在 Java 中,鎖優(yōu)化通常由 JVM 在運行時自動進(jìn)行,但是我們也可以通過代碼設(shè)計來促進(jìn)鎖優(yōu)化。如以下代碼:

class OptimizedLock {

private final Object lock = new Object();

public void method() {

synchronized(lock) {

// 重復(fù)代碼

}

}

}

在這個例子中,我們在方法內(nèi)部加了一個 synchronized 塊。當(dāng)這個方法被頻繁調(diào)用時,JVM 會進(jìn)行鎖優(yōu)化,將多次對同一對象的鎖請求合并為一次。

鎖粗化

與鎖優(yōu)化相反,鎖粗化是將多次獲取同一把鎖的操作合并為一次,也就是將鎖的范圍擴(kuò)大,從而減少獲取鎖的次數(shù),提高性能。如以下代碼:

class CoarseLock {

private final ReentrantLock lock = new ReentrantLock();

public void method() {

lock.lock();

try {

// 代碼塊1

// 代碼塊2

// 代碼塊3

} finally {

lock.unlock();

}

}

}

在這個例子中,我們通過 ReentrantLock 將鎖的范圍擴(kuò)大到整個方法,減少了獲取和釋放鎖的次數(shù)。

6.1. JDK 工具鎖分析工具

JConsole

JConsole 是 JDK 自帶的 Java 監(jiān)控和管理工具,它可以幫助我們分析程序的執(zhí)行情況,包括鎖的使用。當(dāng)我們的程序運行時,可以通過 JConsole 的界面,查看每個線程的狀態(tài),包括它們所持有的鎖。

Java Flight Recorder

Java Flight Recorder 是一個強(qiáng)大的診斷工具,它可以收集和分析 JVM 和應(yīng)用程序的詳細(xì)信息。Java Flight Recorder 可以幫助我們發(fā)現(xiàn)潛在的并發(fā)問題,例如鎖競爭。

本站聲明: 本文章由作者或相關(guān)機(jī)構(gòu)授權(quán)發(fā)布,目的在于傳遞更多信息,并不代表本站贊同其觀點,本站亦不保證或承諾內(nèi)容真實性等。需要轉(zhuǎn)載請聯(lián)系該專欄作者,如若文章內(nèi)容侵犯您的權(quán)益,請及時聯(lián)系本站刪除。
換一批
延伸閱讀

9月2日消息,不造車的華為或?qū)⒋呱龈蟮莫毥谦F公司,隨著阿維塔和賽力斯的入局,華為引望愈發(fā)顯得引人矚目。

關(guān)鍵字: 阿維塔 塞力斯 華為

加利福尼亞州圣克拉拉縣2024年8月30日 /美通社/ -- 數(shù)字化轉(zhuǎn)型技術(shù)解決方案公司Trianz今天宣布,該公司與Amazon Web Services (AWS)簽訂了...

關(guān)鍵字: AWS AN BSP 數(shù)字化

倫敦2024年8月29日 /美通社/ -- 英國汽車技術(shù)公司SODA.Auto推出其旗艦產(chǎn)品SODA V,這是全球首款涵蓋汽車工程師從創(chuàng)意到認(rèn)證的所有需求的工具,可用于創(chuàng)建軟件定義汽車。 SODA V工具的開發(fā)耗時1.5...

關(guān)鍵字: 汽車 人工智能 智能驅(qū)動 BSP

北京2024年8月28日 /美通社/ -- 越來越多用戶希望企業(yè)業(yè)務(wù)能7×24不間斷運行,同時企業(yè)卻面臨越來越多業(yè)務(wù)中斷的風(fēng)險,如企業(yè)系統(tǒng)復(fù)雜性的增加,頻繁的功能更新和發(fā)布等。如何確保業(yè)務(wù)連續(xù)性,提升韌性,成...

關(guān)鍵字: 亞馬遜 解密 控制平面 BSP

8月30日消息,據(jù)媒體報道,騰訊和網(wǎng)易近期正在縮減他們對日本游戲市場的投資。

關(guān)鍵字: 騰訊 編碼器 CPU

8月28日消息,今天上午,2024中國國際大數(shù)據(jù)產(chǎn)業(yè)博覽會開幕式在貴陽舉行,華為董事、質(zhì)量流程IT總裁陶景文發(fā)表了演講。

關(guān)鍵字: 華為 12nm EDA 半導(dǎo)體

8月28日消息,在2024中國國際大數(shù)據(jù)產(chǎn)業(yè)博覽會上,華為常務(wù)董事、華為云CEO張平安發(fā)表演講稱,數(shù)字世界的話語權(quán)最終是由生態(tài)的繁榮決定的。

關(guān)鍵字: 華為 12nm 手機(jī) 衛(wèi)星通信

要點: 有效應(yīng)對環(huán)境變化,經(jīng)營業(yè)績穩(wěn)中有升 落實提質(zhì)增效舉措,毛利潤率延續(xù)升勢 戰(zhàn)略布局成效顯著,戰(zhàn)新業(yè)務(wù)引領(lǐng)增長 以科技創(chuàng)新為引領(lǐng),提升企業(yè)核心競爭力 堅持高質(zhì)量發(fā)展策略,塑強(qiáng)核心競爭優(yōu)勢...

關(guān)鍵字: 通信 BSP 電信運營商 數(shù)字經(jīng)濟(jì)

北京2024年8月27日 /美通社/ -- 8月21日,由中央廣播電視總臺與中國電影電視技術(shù)學(xué)會聯(lián)合牽頭組建的NVI技術(shù)創(chuàng)新聯(lián)盟在BIRTV2024超高清全產(chǎn)業(yè)鏈發(fā)展研討會上宣布正式成立。 活動現(xiàn)場 NVI技術(shù)創(chuàng)新聯(lián)...

關(guān)鍵字: VI 傳輸協(xié)議 音頻 BSP

北京2024年8月27日 /美通社/ -- 在8月23日舉辦的2024年長三角生態(tài)綠色一體化發(fā)展示范區(qū)聯(lián)合招商會上,軟通動力信息技術(shù)(集團(tuán))股份有限公司(以下簡稱"軟通動力")與長三角投資(上海)有限...

關(guān)鍵字: BSP 信息技術(shù)
關(guān)閉