線程中sleep()與wait()區(qū)別
sleep是讓線程暫停指定的時間,wait是讓線程等待,知道有notify方法出現(xiàn)在繼續(xù)運行,sleep不釋放對象鎖,wait釋放對象鎖。這在很多地方都有寫到,但沒有例子不好理解
import java.util.ArrayList; import java.util.List; public class thread { public List list=new ArrayList(); Integer i=0;//新建一個對象,隨便什么,就為了借用它的鎖而已。 public static void main(String[] args){ thread th=new thread(); A a=th.new A(); B b=th.new B(); Thread add=new Thread(a); Thread get=new Thread(b); add.start(); get.start(); } class A extends Thread{ public void run(){ synchronized(i){ try { //Thread.sleep(1000); i.wait();//這里調(diào)用了對象i的wait方法,那么就表示所在線程暫時放棄i的對象鎖,并阻塞在這里,同時把機(jī)會讓給同樣使用i鎖的B線程執(zhí)行 } catch (InterruptedException e) { // TODO Auto-generated catch block e.printStackTrace(); } for(int j=0;j
從如上代碼中可見,線程B是負(fù)責(zé)向集合中加值的,線程A是負(fù)責(zé)從集合中取值的,如果不進(jìn)行同步處理,那么如果兩個線程同時運行,那么A可能在B加入值之前就從集合中取值,那么就會報錯,所以必須進(jìn)行同步代碼的操作(也可以加入flag來判斷是否到了A中從集合里取值的時機(jī))。
而用sleep就沒有上面的功能了,因為從代碼中可以發(fā)現(xiàn),不管是A線程先搶到cpu開始執(zhí)行還是B線程先執(zhí)行,他們中的一個肯定會拿到i的鎖的,但拿到后i的鎖就沒有使用wait方法釋放過,那假設(shè)B先拿到鎖了,向list中加入值完成了,那也別指望A能打印出來,因為A一直阻塞在(i){