java - 線程同步為什么不一樣
問題描述
package com.dome;
public class Thread01 {
private volatile static int a =10;Thread td1 = new Thread(){public void run(){for(int i=0;i<3;i++){ a = a+1;System.out.println(i+'td1:='+a);} } };Thread td2 = new Thread(){ public void run(){for(int i=0;i<3;i++){ a -=1; System.out.println(i+'td2:='+a);} } };public static void main(String[] args) { Thread01 th = new Thread01(); th.td1.start();th.td2.start(); }
}
0td1:=90td2:=91td1:=101td2:=92td1:=102td2:=9
問題解答
回答1:a = a + 1, a = a - 1 這樣的語句,事實上涉及了 讀取-修改-寫入 三個操作:
讀取變量到棧中某個位置
對棧中該位置的值進行加 (減)1
將自增后的值寫回到變量對應的存儲位置
因此雖然變量 a 使用 volatile 修飾,但并不能使涉及上面三個操作的 a = a + 1,a = a - 1具有原子性。為了保證同步性,需要使用 synchronized:
public class Thread01 { private volatile static int a = 10; Thread td1 = new Thread() {public void run() { for (int i = 0; i < 3; i++) {synchronized (Thread01.class) { a = a + 1; System.out.println(i + 'td1:=' + a);} }} }; Thread td2 = new Thread() {public void run() { for (int i = 0; i < 3; i++) {synchronized (Thread01.class) { a -= 1; System.out.println(i + 'td2:=' + a);} }} }; public static void main(String[] args) {Thread01 th = new Thread01();th.td1.start();th.td2.start(); }}
某次運行結果:
(td1 出現的地方,a 就 +1;td2 出現的地方,a 就 -1)
相關文章:
1. Span標簽2. css - 求推薦適用于vue2的框架 像bootstrap這種類型的3. docker-machine添加一個已有的docker主機問題4. css - 關于div自適應問題,大家看圖吧,說不清5. 關docker hub上有些鏡像的tag被標記““This image has vulnerabilities””6. SessionNotFoundException:會話ID為null。調用quit()后使用WebDriver嗎?(硒)7. android新手一枚,android使用httclient獲取服務器端數據失敗,但是用java工程運行就可以成功獲取。8. angular.js使用$resource服務把數據存入mongodb的問題。9. java - Collections類里的swap函數,源碼為什么要新定義一個final的List型變量l指向傳入的list?10. python - django如何每次調用標簽的時候都取隨機數據
