Java多線程同步Synchronized使用分析

同步的概念:

同步分為 同步方法 和 同步塊 兩種方式。

鎖定的內(nèi)容分為 鎖定類的某個(gè)特定實(shí)例 和 鎖定類對(duì)象(類的所有實(shí)例)

變量分為 實(shí)例變量(不帶static的變量) 和 類變量(帶static的變量)

使用同步的原因

1. 在系統(tǒng)中對(duì)訪類要使用多線程進(jìn)行訪問;

2. 在該類中有 類變量, 或者是 在類的方法中有訪問 公共資源(如一個(gè)外部文件的讀寫)。

同步鎖鎖定的內(nèi)容是什么?

無論你將Synchronized加在方法前還是加在一個(gè)變量前,其鎖定的都是一個(gè) 類對(duì)象。 每一個(gè)對(duì)象都只有一個(gè)鎖與之相關(guān)聯(lián)。

下例中分情況的列舉各種情況下的同步效果

1. Synchronized 加在方法上, (同步方法,鎖定類實(shí)例)

Java代碼

publicclassDemo1?{

publicsynchronizedvoidm1(){

//...............

}

publicvoidm2(){

//............

synchronized(this){

//.........

}

//........

}

}

這兩種寫法的效果是一樣的,鎖定的都是類實(shí)例對(duì)象。如果有一個(gè) 類實(shí)例對(duì)象: demo = new Demo1(),另外有兩個(gè)線程: thread1,thread2,都調(diào)用了demo 對(duì)象,那么,在同一時(shí)間,如果 thread1調(diào)用了demo.m1(),則thread2在該時(shí)間內(nèi)不能訪問demo.m1() 和 demo.m2(); 因?yàn)閠hread1把demo這個(gè)對(duì)象的鎖使用了,所以無法分給其它線程使用

但是,如果thread1調(diào)用 demo1.m1(), thread2調(diào)用 demo2.m1(), 則可以同時(shí)進(jìn)行,因?yàn)樗鼈冋{(diào)用的是不同的Demo1類對(duì)象實(shí)例。

2. Synchronized 加在變量上, (同步塊,鎖定類實(shí)例)

Java代碼

publicclassDemo2?{

Object?a?=newObject();

Object?b?=newObject();

publicvoidm1(){

//............

synchronized(a){

//.........

}

//........

}

publicvoidm2(){

//............

synchronized(b){

//.........

}

//........

}

}

這種情況下,是實(shí)現(xiàn)代碼塊鎖定,鎖定的對(duì)象是 變量 a 或 b; (注意,a 、b 都是非static 的)如果有一個(gè) 類實(shí)例對(duì)象: demo = new Demo2(),另外有兩個(gè)線程: thread1,thread2,都調(diào)用了demo 對(duì)象,那么,在同一時(shí)間,如果 thread1調(diào)用了demo.m1(),則thread2在該時(shí)間內(nèi)可以訪問demo.m2();但不能訪問 demo.m1() 的同步塊, 因?yàn)閍被 thread1鎖定了。

3. Synchronized 鎖定的是 類變量 ,即static 變量(可能是屬性,可能是方法)(鎖定類對(duì)象)

Java代碼

publicclassDemo3?{

staticObject?o?=newObject();

publicstaticsynchronizedvoidm1()?{

//....

}

publicstaticvoidm2()?{

//...

synchronized(Demo3.class)?{

//.....

}

//.....

}

publicstaticvoidm3()?{

//..........

try{

synchronized(Class.forName("Demo3"))?{

//............

}

}catch(ClassNotFoundException?ex)?{

}

//.............

}

publicstaticvoidm4()?{

//............

synchronized(o){

//........

}

//..........

}

}

以上4個(gè)方法中實(shí)現(xiàn)的效果都是一樣的,其鎖定的對(duì)象都是類Demo3,而不是類實(shí)例對(duì)象 ,即在多線程中,其共享的資源是屬于類的,而不是屬于類對(duì)象的。在這種情況下,如果thread1 訪問了這4個(gè)方法中的任何一個(gè), 在同一時(shí)間內(nèi)其它的線程都不能訪問 這4個(gè)方法。

4. 類的方法中訪問了多線程共同的資源, 且該資源是可變的,這種情況下也是需要進(jìn)行同步的

Java代碼

publicclassDemo4?{

staticString?path?="file?path";

publicvoidreadConfiFile()?{

synchronized(path)?{

//?讀取該path指定的文件。

}

}

publicvoidwriteConfiFile()?{

synchronized(path)?{

//寫信息到該path指定的文件。

}

}

}

這種情況下,必須鎖定為 類變量,而不能進(jìn)行鎖定類實(shí)例對(duì)象,因?yàn)檫@是變象的一種類資源共享,而不是類實(shí)例對(duì)象資源共享。

線程,成也其,敗也其,用好了可以提升性能,用不好則會(huì)使系統(tǒng)后患無窮。

PS: 進(jìn)行線程同步需要很大的系統(tǒng)開銷, 所以,在使用時(shí),如果不是必須的,則盡量不使用同步功能。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡(jiǎn)書系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。

推薦閱讀更多精彩內(nèi)容

  • 本文主要講了java中多線程的使用方法、線程同步、線程數(shù)據(jù)傳遞、線程狀態(tài)及相應(yīng)的一些線程函數(shù)用法、概述等。 首先講...
    李欣陽閱讀 2,493評(píng)論 1 15
  • 1. Java基礎(chǔ)部分 基礎(chǔ)部分的順序:基本語法,類相關(guān)的語法,內(nèi)部類的語法,繼承相關(guān)的語法,異常的語法,線程的語...
    子非魚_t_閱讀 31,765評(píng)論 18 399
  • Java多線程學(xué)習(xí) [-] 一擴(kuò)展javalangThread類 二實(shí)現(xiàn)javalangRunnable接口 三T...
    影馳閱讀 2,987評(píng)論 1 18
  • 寫在前面的話: 這篇博客是我從這里“轉(zhuǎn)載”的,為什么轉(zhuǎn)載兩個(gè)字加“”呢?因?yàn)檫@絕不是簡(jiǎn)單的復(fù)制粘貼,我花了五六個(gè)小...
    SmartSean閱讀 4,791評(píng)論 12 45
  • 我們?cè)谛扌械倪^程中,總是苦于沒有老師的指導(dǎo)。實(shí)際上,我們每天詠念的《心經(jīng)》,就是我們修行、生活最好的指導(dǎo),最親切的...
    旺財(cái)楊子閱讀 2,252評(píng)論 2 50