同步的概念:
同步分為 同步方法 和 同步塊 兩種方式揪漩。
鎖定的內(nèi)容分為 鎖定類的某個特定實例 和 鎖定類對象(類的所有實例)
變量分為 實例變量(不帶static的變量) 和 類變量(帶static的變量)
使用同步的原因
1. 在系統(tǒng)中對訪類要使用多線程進行訪問;
2. 在該類中有 類變量, 或者是 在類的方法中有訪問 公共資源(如一個外部文件的讀寫)措近。
同步鎖鎖定的內(nèi)容是什么?
無論你將Synchronized加在方法前還是加在一個變量前熊昌,其鎖定的都是一個 類對象。 每一個對象都只有一個鎖與之相關(guān)聯(lián)憨颠。
下例中分情況的列舉各種情況下的同步效果
1. Synchronized 加在方法上胳徽, (同步方法,鎖定類實例)
Java代碼
publicclassDemo1?{
publicsynchronizedvoidm1(){
//...............
}
publicvoidm2(){
//............
synchronized(this){
//.........
}
//........
}
}
這兩種寫法的效果是一樣的爽彤,鎖定的都是類實例對象养盗。如果有一個 類實例對象: demo = new Demo1(),另外有兩個線程: thread1适篙,thread2往核,都調(diào)用了demo 對象,那么嚷节,在同一時間聂儒,如果 thread1調(diào)用了demo.m1()虎锚,則thread2在該時間內(nèi)不能訪問demo.m1() 和 demo.m2(); 因為thread1把demo這個對象的鎖使用了,所以無法分給其它線程使用
但是衩婚,如果thread1調(diào)用 demo1.m1(), thread2調(diào)用 demo2.m1(), 則可以同時進行窜护,因為它們調(diào)用的是不同的Demo1類對象實例。
2. Synchronized 加在變量上谅猾, (同步塊柄慰,鎖定類實例)
Java代碼
publicclassDemo2?{
Object?a?=newObject();
Object?b?=newObject();
publicvoidm1(){
//............
synchronized(a){
//.........
}
//........
}
publicvoidm2(){
//............
synchronized(b){
//.........
}
//........
}
}
這種情況下,是實現(xiàn)代碼塊鎖定税娜,鎖定的對象是 變量 a 或 b; (注意坐搔,a 、b 都是非static 的)如果有一個 類實例對象: demo = new Demo2()敬矩,另外有兩個線程: thread1概行,thread2,都調(diào)用了demo 對象弧岳,那么凳忙,在同一時間,如果 thread1調(diào)用了demo.m1()禽炬,則thread2在該時間內(nèi)可以訪問demo.m2();但不能訪問 demo.m1() 的同步塊涧卵, 因為a被 thread1鎖定了。
3. Synchronized 鎖定的是 類變量 腹尖,即static 變量(可能是屬性柳恐,可能是方法)(鎖定類對象)
Java代碼
publicclassDemo3?{
staticObject?o?=newObject();
publicstaticsynchronizedvoidm1()?{
//....
}
publicstaticvoidm2()?{
//...
synchronized(Demo3.class)?{
//.....
}
//.....
}
publicstaticvoidm3()?{
//..........
try{
synchronized(Class.forName("Demo3"))?{
//............
}
}catch(ClassNotFoundException?ex)?{
}
//.............
}
publicstaticvoidm4()?{
//............
synchronized(o){
//........
}
//..........
}
}
以上4個方法中實現(xiàn)的效果都是一樣的,其鎖定的對象都是類Demo3热幔,而不是類實例對象 乐设,即在多線程中,其共享的資源是屬于類的绎巨,而不是屬于類對象的近尚。在這種情況下,如果thread1 訪問了這4個方法中的任何一個场勤, 在同一時間內(nèi)其它的線程都不能訪問 這4個方法戈锻。
4. 類的方法中訪問了多線程共同的資源, 且該資源是可變的,這種情況下也是需要進行同步的
Java代碼
publicclassDemo4?{
staticString?path?="file?path";
publicvoidreadConfiFile()?{
synchronized(path)?{
//?讀取該path指定的文件和媳。
}
}
publicvoidwriteConfiFile()?{
synchronized(path)?{
//寫信息到該path指定的文件格遭。
}
}
}
這種情況下,必須鎖定為 類變量窗价,而不能進行鎖定類實例對象如庭,因為這是變象的一種類資源共享叹卷,而不是類實例對象資源共享撼港。
線程坪它,成也其,敗也其帝牡,用好了可以提升性能往毡,用不好則會使系統(tǒng)后患無窮。
PS: 進行線程同步需要很大的系統(tǒng)開銷靶溜, 所以开瞭,在使用時,如果不是必須的罩息,則盡量不使用同步功能嗤详。