溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

Java中怎么利用Synchronized實(shí)現(xiàn)多線程同步

發(fā)布時(shí)間:2021-07-01 17:00:26 來源:億速云 閱讀:156 作者:Leah 欄目:編程語言

這期內(nèi)容當(dāng)中小編將會(huì)給大家?guī)碛嘘P(guān)Java中怎么利用Synchronized實(shí)現(xiàn)多線程同步,文章內(nèi)容豐富且以專業(yè)的角度為大家分析和敘述,閱讀完這篇文章希望大家可以有所收獲。

使用同步的原因

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代碼

public class Demo1 {             public synchronized void m1(){             //...............         }             public void m2(){             //............                 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代碼

public class Demo2 {         Object a = new Object();         Object b = new Object();             public void m1(){             //............                 synchronized(a){                 //.........             }                 //........         }             public void m2(){             //............                 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代碼

public class Demo3 {         static Object o = new Object();             public static synchronized void m1() {             //....         }             public static void m2() {             //...             synchronized (Demo3.class) {                 //.....             }             //.....         }             public static void m3() {             //..........             try {                 synchronized (Class.forName("Demo3")) {                   //............                 }             } catch (ClassNotFoundException ex) {             }             //.............         }             public static void m4() {             //............            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代碼

public class Demo4 {         static String path = "file path";             public void readConfiFile() {             synchronized (path) {                // 讀取該path指定的文件。             }             }             public void writeConfiFile() {             synchronized (path) {                 //寫信息到該path指定的文件。             }         }     }

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

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

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

上述就是小編為大家分享的Java中怎么利用Synchronized實(shí)現(xiàn)多線程同步了,如果剛好有類似的疑惑,不妨參照上述分析進(jìn)行理解。如果想知道更多相關(guān)知識(shí),歡迎關(guān)注億速云行業(yè)資訊頻道。

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI