線程的同步和鎖的問題,我們先來看個例子,
下面的例子我們希望兩個線程對m變數進行削減
| 代碼如下 |
複製代碼 |
packagecom.javaer.thread; publicclassSysTestimplementsRunnable{ intm=10; publicstaticvoidmain(String[]args){ SysTests1=newSysTest(); Threadt1=newThread(s1,"Thread-t1"); Threadt2=newThread(s1,"Thread-t2"); t1.start(); t2.start(); } publicvoidreduce(){ m--; } @Override publicvoidrun(){ while(m>0) { try{ Thread.sleep(1); }catch(InterruptedExceptione){ //TODOAuto-generatedcatchblock e.printStackTrace(); } this.reduce(); System.out.println(Thread.currentThread().getName()+"m的值:"+m); } } } |
Thread-t1m的值:8
Thread-t2m的值:8
Thread-t1m的值:7
Thread-t2m的值:6
Thread-t1m的值:5
Thread-t2m的值:4
Thread-t1m的值:3
Thread-t2m的值:2
Thread-t1m的值:0
Thread-t2m的值:0
兩個線程不加控制的執行了reduce操作,於是出現了兩個8的情況。
如果我們想線程按照一定的順序執行,可以進行同步
同步哪些修改變數的代碼,使用synchronized關鍵字同步方法或代碼。
這裡我們對reduce操作進行鎖
publicsynchronizedvoidreduce()
即可實現我們想要的。
java同步鎖的原理
一個對象只有一個鎖。所以,如果一個線程獲得該鎖,就沒有其他線程可以獲得鎖,直到第一個線程釋放(或返回)鎖。這也意味著任何其他線程都不能進入該對象上的synchronized方法或代碼塊,直到該鎖被釋放。
同步和鎖的要點
1.只能同步方法,不能同步變數和類
2.不要同步所有方法,僅僅同步必要的方法
3.如果一個synchronized方法正在被一個線程調用,其他線程不能調用,進入等待狀態,只要方法被釋放。
4.線程休眠的時候,不會自動釋放鎖
5.同步會傷害效率,多線程的目的是並發,鎖的目的是反並發。所以儘可能的縮小鎖的範圍。
6對於非靜態欄位中可更改的資料,通常使用非靜態方法訪問。
對於靜態欄位中可更改的資料,通常使用靜態方法訪問。
什麼時候需要用同步
1.訪問依賴的資源,比如都要從1個任務池裡領取任務的時候。
2.多個線程都會修改同一個公開變數的時候
死結
當一個佔用了鎖的線程,出了問題,比如崩潰。其他等待的線程全部發生阻塞,將發生死結,一旦發生死結,程式就死掉。