這篇文章主要講解了“Java多線程編程中的鎖有什么用”,文中的講解內(nèi)容簡單清晰,易于學(xué)習(xí)與理解,下面請大家跟著小編的思路慢慢深入,一起來研究和學(xué)習(xí)“Java多線程編程中的鎖有什么用”吧!
成都創(chuàng)新互聯(lián)主要從事成都網(wǎng)站設(shè)計、成都做網(wǎng)站、網(wǎng)頁設(shè)計、企業(yè)做網(wǎng)站、公司建網(wǎng)站等業(yè)務(wù)。立足成都服務(wù)黑河,10余年網(wǎng)站建設(shè)經(jīng)驗,價格優(yōu)惠、服務(wù)專業(yè),歡迎來電咨詢建站服務(wù):028-86922220
閱讀目錄
一、盡量不要鎖住方法
二、縮小同步代碼塊,只鎖數(shù)據(jù)
三、鎖中盡量不要再包含鎖
四、將鎖私有化,在內(nèi)部管理鎖
五、進(jìn)行適當(dāng)?shù)逆i分解
正文
并發(fā)環(huán)境下進(jìn)行編程時,需要使用鎖機(jī)制來同步多線程間的操作,保證共享資源的互斥訪問。加鎖會帶來性能上的損壞,似乎是眾所周知的事情。然而,加鎖本身不會帶來多少的性能消耗,性能主要是在線程的獲取鎖的過程。
如果只有一個線程競爭鎖,此時并不存在多線程競爭的情況,那么JVM會進(jìn)行優(yōu)化,那么這時加鎖帶來的性能消耗基本可以忽略。因此,規(guī)范加鎖的操作,優(yōu)化鎖的使用方法,避免不必要的線程競爭,不僅可以提高程序性能,也能避免不規(guī)范加鎖可能造成線程死鎖問題,提高程序健壯性。下面闡述幾種鎖優(yōu)化的思路。
一、盡量不要鎖住方法
在普通成員函數(shù)上加鎖時,線程獲得的是該方法所在對象的對象鎖。此時整個對象都會被鎖住。這也意味著,如果這個對象提供的多個同步方法是針對不同業(yè)務(wù)的,那么由于整個對象被鎖住,一個業(yè)務(wù)業(yè)務(wù)在處理時,其他不相關(guān)的業(yè)務(wù)線程也必須wait。下面的例子展示了這種情況:
LockMethod類包含兩個同步方法,分別在兩種業(yè)務(wù)處理中被調(diào)用:
public class LockMethod { public synchronized void busiA() { for (int i = 0; i < 10000; i++) { System.out.println(Thread.currentThread().getName() + "deal with bussiness A:"+i); } } public synchronized void busiB() { for (int i = 0; i < 10000; i++) { System.out.println(Thread.currentThread().getName() + "deal with bussiness B:"+i); } } }
BUSSA是線程類,用來處理A業(yè)務(wù),調(diào)用的是LockMethod的busiA()方法:
public class BUSSA extends Thread { LockMethod lockMethod; void deal(LockMethod lockMethod){ this.lockMethod = lockMethod; } @Override public void run() { super.run(); lockMethod.busiA(); } }
BUSSB是線程類,用來處理B業(yè)務(wù),調(diào)用的是LockMethod的busiB()方法:
public class BUSSB extends Thread { LockMethod lockMethod; void deal(LockMethod lockMethod){ this.lockMethod = lockMethod; } @Override public void run() { super.run(); lockMethod.busiB(); } }
TestLockMethod類,使用線程BUSSA與BUSSB進(jìn)行業(yè)務(wù)處理:
public class TestLockMethod extends Thread { public static void main(String[] args) { LockMethod lockMethod = new LockMethod(); BUSSA bussa = new BUSSA(); BUSSB bussb = new BUSSB(); bussa.deal(lockMethod); bussb.deal(lockMethod); bussa.start(); bussb.start(); } } 運(yùn)行程
運(yùn)行程序,可以看到在線程bussa 執(zhí)行的過程中,bussb是不能夠進(jìn)入函數(shù) busiB()的,因為此時lockMethod 的對象鎖被線程bussa獲取了。
二、縮小同步代碼塊,只鎖數(shù)據(jù)
有時候為了編程方便,有些人會synchnoized很大的一塊代碼,如果這個代碼塊中的某些操作與共享資源并不相關(guān),那么應(yīng)當(dāng)把它們放到同步塊外部,避免長時間的持有鎖,造成其他線程一直處于等待狀態(tài)。尤其是一些循環(huán)操作、同步I/O操作。
不止是在代碼的行數(shù)范圍上縮小同步塊,在執(zhí)行邏輯上,也應(yīng)該縮小同步塊,例如多加一些條件判斷,符合條件的再進(jìn)行同步,而不是同步之后再進(jìn)行條件判斷,盡量減少不必要的進(jìn)入同步塊的邏輯。
三、鎖中盡量不要再包含鎖
這種情況經(jīng)常發(fā)生,線程在得到了A鎖之后,在同步方法塊中調(diào)用了另外對象的同步方法,獲得了第二個鎖,這樣可能導(dǎo)致一個調(diào)用堆棧中有多把鎖的請求,多線程情況下可能會出現(xiàn)很復(fù)雜、難以分析的異常情況,導(dǎo)致死鎖的發(fā)生。下面的代碼顯示了這種情況:
synchronized(A){ synchronized(B){ } }
或是在同步塊中調(diào)用了同步方法:
synchronized(A){ B b = objArrayList.get(0); b.method(); //這是一個同步方法 }
解決的辦法是跳出來加鎖,不要包含加鎖:
{ B b = null; synchronized(A){ b = objArrayList.get(0); } b.method(); }
四、將鎖私有化,在內(nèi)部管理鎖
把鎖作為一個私有的對象,外部不能拿到這個對象,更安全一些。對象可能被其他線程直接進(jìn)行加鎖操作,此時線程便持有了該對象的對象鎖,例如下面這種情況:
class A { public void method1() { } } class B { public void method1() { A a = new A(); synchronized (a) { //直接進(jìn)行加鎖 a.method1(); } } }
這種使用方式下,對象a的對象鎖被外部所持有,讓這把鎖在外部多個地方被使用是比較危險的,對代碼的邏輯流程閱讀也造成困擾。一種更好的方式是在類的內(nèi)部自己管理鎖,外部需要同步方案時,也是通過接口方式來提供同步操作:
class A { private Object lock = new Object(); public void method1() { synchronized (lock){ } } } class B { public void method1() { A a = new A(); a.method1(); } }
五、進(jìn)行適當(dāng)?shù)逆i分解
考慮下面這段程序:
public class GameServer { public Map<String, List<Player>> tables = new HashMap<String, List<Player>>(); public void join(Player player, Table table) { if (player.getAccountBalance() > table.getLimit()) { synchronized (tables) { List<Player> tablePlayers = tables.get(table.getId()); if (tablePlayers.size() < 9) { tablePlayers.add(player); } } } } public void leave(Player player, Table table) {/*省略*/} public void createTable() {/*省略*/} public void destroyTable(Table table) {/*省略*/} }
在這個例子中,join方法只使用一個同步鎖,來獲取tables中的List對象,然后判斷玩家數(shù)量是不是小于9,如果是,就調(diào)增加一個玩家。當(dāng)有成千上萬個List存在tables中時,對tables鎖的競爭將非常激烈。
在這里,我們可以考慮進(jìn)行鎖的分解:快速取出數(shù)據(jù)之后,對List對象進(jìn)行加鎖,讓其他線程可快速競爭獲得tables對象鎖:
public class GameServer { public Map<String, List<Player>> tables = new HashMap<String, List<Player>>(); public void join(Player player, Table table) { if (player.getAccountBalance() > table.getLimit()) { List<Player> tablePlayers = null; synchronized (tables) { tablePlayers = tables.get(table.getId()); } synchronized (tablePlayers) { if (tablePlayers.size() < 9) { tablePlayers.add(player); } } } } public void leave(Player player, Table table) {/*省略*/} public void createTable() {/*省略*/} public void destroyTable(Table table) {/*省略*/} }
感謝各位的閱讀,以上就是“Java多線程編程中的鎖有什么用”的內(nèi)容了,經(jīng)過本文的學(xué)習(xí)后,相信大家對Java多線程編程中的鎖有什么用這一問題有了更深刻的體會,具體使用情況還需要大家實踐驗證。這里是創(chuàng)新互聯(lián),小編將為大家推送更多相關(guān)知識點的文章,歡迎關(guān)注!
分享題目:Java多線程編程中的鎖有什么用
文章地址:http://m.kartarina.com/article20/jedjjo.html
成都網(wǎng)站建設(shè)公司_創(chuàng)新互聯(lián),為您提供網(wǎng)站排名、標(biāo)簽優(yōu)化、做網(wǎng)站、Google、虛擬主機(jī)、面包屑導(dǎo)航
聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶投稿、用戶轉(zhuǎn)載內(nèi)容為主,如果涉及侵權(quán)請盡快告知,我們將會在第一時間刪除。文章觀點不代表本網(wǎng)站立場,如需處理請聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時需注明來源: 創(chuàng)新互聯(lián)