亚洲视频二区_亚洲欧洲日本天天堂在线观看_日韩一区二区在线观看_中文字幕不卡一区

公告:魔扣目錄網為廣大站長提供免費收錄網站服務,提交前請做好本站友鏈:【 網站目錄:http://www.430618.com 】, 免友鏈快審服務(50元/站),

點擊這里在線咨詢客服
新站提交
  • 網站:51998
  • 待審:31
  • 小程序:12
  • 文章:1030137
  • 會員:747



并發環境下進行編程時,需要使用鎖機制來同步多線程間的操作,保證共享資源的互斥訪問。加鎖會帶來性能上的損壞,似乎是眾所周知的事情。然而,加鎖本身不會帶來多少的性能消耗,性能主要是在線程的獲取鎖的過程。如果只有一個線程競爭鎖,此時并不存在多線程競爭的情況,那么JVM會進行優化,那么這時加鎖帶來的性能消耗基本可以忽略。因此,規范加鎖的操作,優化鎖的使用方法,避免不必要的線程競爭,不僅可以提高程序性能,也能避免不規范加鎖可能造成線程死鎖問題,提高程序健壯性。下面闡述幾種鎖優化的思路。

一、盡量不要鎖住方法

在普通成員函數上加鎖時,線程獲得的是該方法所在對象的對象鎖。此時整個對象都會被鎖住。這也意味著,如果這個對象提供的多個同步方法是針對不同業務的,那么由于整個對象被鎖住,一個業務業務在處理時,其他不相關的業務線程也必須wait。下面的例子展示了這種情況:

LockMethod類包含兩個同步方法,分別在兩種業務處理中被調用:

 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業務,調用的是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業務,調用的是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進行業務處理:

 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();
}
} 

運行程序,可以看到在線程bussa 執行的過程中,bussb是不能夠進入函數 busiB()的,因為此時lockMethod 的對象鎖被線程bussa獲取了。

二、縮小同步代碼塊,只鎖數據

有時候為了編程方便,有些人會synchnoized很大的一塊代碼,如果這個代碼塊中的某些操作與共享資源并不相關,那么應當把它們放到同步塊外部,避免長時間的持有鎖,造成其他線程一直處于等待狀態。尤其是一些循環操作、同步I/O操作。不止是在代碼的行數范圍上縮小同步塊,在執行邏輯上,也應該縮小同步塊,例如多加一些條件判斷,符合條件的再進行同步,而不是同步之后再進行條件判斷,盡量減少不必要的進入同步塊的邏輯。

三、鎖中盡量不要再包含鎖

這種情況經常發生,線程在得到了A鎖之后,在同步方法塊中調用了另外對象的同步方法,獲得了第二個鎖,這樣可能導致一個調用堆棧中有多把鎖的請求,多線程情況下可能會出現很復雜、難以分析的異常情況,導致死鎖的發生。下面的代碼顯示了這種情況:

synchronized(A){
synchronized(B){
 } 
1
2
}

或是在同步塊中調用了同步方法:

synchronized(A){
B b = objArrayList.get(0);
b.method(); //這是一個同步方法
}

解決的辦法是跳出來加鎖,不要包含加鎖:

 { B b = null;
synchronized(A)
{ b = objArrayList.get(0); } b.method(); 
}

四、將鎖私有化,在內部管理鎖

把鎖作為一個私有的對象,外部不能拿到這個對象,更安全一些。對象可能被其他線程直接進行加鎖操作,此時線程便持有了該對象的對象鎖,例如下面這種情況:

 class A { public void method1() { } }
class B { public void method1() { 
A a = new A(); synchronized (a) { //直接進行加鎖       
a.method1();
 }
}
}

這種使用方式下,對象a的對象鎖被外部所持有,讓這把鎖在外部多個地方被使用是比較危險的,對代碼的邏輯流程閱讀也造成困擾。一種更好的方式是在類的內部自己管理鎖,外部需要同步方案時,也是通過接口方式來提供同步操作:

class A { private Object lock = new Object(); 
public void method1() { 
synchronized (lock){
 }
}
}
class B { public void method1() { A a = new A(); a.method1(); } }
 以上就是我的分享,覺得有收獲的朋友們可以點個關注收藏轉發一下哦,感謝支持!

分享到:
標簽:多線程 編程 Java
用戶無頭像

網友整理

注冊時間:

網站:5 個   小程序:0 個  文章:12 篇

  • 51998

    網站

  • 12

    小程序

  • 1030137

    文章

  • 747

    會員

趕快注冊賬號,推廣您的網站吧!
最新入駐小程序

數獨大挑戰2018-06-03

數獨一種數學游戲,玩家需要根據9

答題星2018-06-03

您可以通過答題星輕松地創建試卷

全階人生考試2018-06-03

各種考試題,題庫,初中,高中,大學四六

運動步數有氧達人2018-06-03

記錄運動步數,積累氧氣值。還可偷

每日養生app2018-06-03

每日養生,天天健康

體育訓練成績評定2018-06-03

通用課目體育訓練成績評定