每天一例多執行緒[day2]-----synchronized與多個執行緒多個鎖
package com.jeff.base.sync002;
-
/**
-
* 多個執行緒多個鎖
-
*
-
* 關鍵字synchronized取得的鎖都是物件鎖,而不是把一段程式碼(方法)當做鎖,
-
* 所以程式碼中哪個執行緒先執行synchronized關鍵字的方法,哪個執行緒就持有該方法所屬物件的鎖(Lock),
-
*
-
* @author jeff
-
*
-
*/
-
public class MultiThread {
-
private static int num = 0;
-
/** static */
-
public static synchronized void printNum(String tag){
-
try {
-
if(tag.equals("a")){
-
num = 100;
-
System.out.println("tag a, set num over!");
-
Thread.sleep(1000);
-
} else {
-
num = 200;
-
System.out.println("tag b, set num over!");
-
}
-
System.out.println("tag " + tag + ", num = " + num);
-
} catch (InterruptedException e) {
-
e.printStackTrace();
-
}
-
}
-
//注意觀察run方法輸出順序
-
public static void main(String[] args) {
-
/**
-
* 倆個不同的物件m1/m2:
-
* 一個物件一把鎖,m1和m2兩個物件獲得的是自己的那一把物件鎖 ,兩者沒有任何關係,不存在同步問題。
-
*
-
* 在靜態型別的printNum方法上加synchronized關鍵字,表示鎖定.class類,類一級別的鎖(獨佔.class類)。
-
* m1和m2兩個執行緒物件在訪問printNum時,訪問的是同一把鎖,
-
* 所以最終的結果一定是:要麼先執行a的結果,再執行b的:
-
* tag a, set num over!
-
tag a, num = 100
-
tag b, set num over!
-
tag b, num = 200
-
要麼是先執行b的,再執行a的:
-
tag b, set num over!
-
tag b, num = 100
-
tag a, set num over!
-
tag a, num = 200
-
*
-
*/
-
final MultiThread m1 = new MultiThread();
-
final MultiThread m2 = new MultiThread();
-
Thread t1 = new Thread(new Runnable() {
-
@Override
-
public void run() {
-
m1.printNum("a");
-
}
-
});
-
Thread t2 = new Thread(new Runnable() {
-
@Override
-
public void run() {
-
m2.printNum("b");
-
}
-
});
-
t1.start();
-
t2.start();
-
}
-
}
如果把printNum方法的static關鍵詞去掉,列印的結果:
-
tag a, set num over!
-
tag b, set num over!
-
tag b, num = 200
-
tag a, num = 200
也就是說,這兩個執行緒之間獲得的是自己的那一把鎖,多個執行緒多個鎖,各自無影響。