wait()、notify()、notifyAll()是三個定義在Object類里的方法,可以用來控制線程的狀態
public final native void notify(); public final native void notifyAll(); public final native void wait(long l) throws InterruptedException; public final void wait(long l, int i) throws InterruptedException { if(l < 0L) throw new IllegalArgumentException("timeout value is negative"); if(i < 0 || i > 999999) throw new IllegalArgumentException("nanosecond timeout value out of range"); if(i > 0) l++; wait(l); } public final void wait() throws InterruptedException { wait(0L); }
這三個方法最終調用的都是jvm級的final native方法。隨着jvm運行平台的不同可能有些許差異。
- 如果對象調用了wait方法就會使持有該對象的線程把該對象的控制權交出去,然后處於等待狀態。
- 如果對象調用了notify方法就會通知某個正在等待這個對象的控制權的線程可以繼續運行。
- 如果對象調用了notifyAll方法就會通知所有等待這個對象控制權的線程繼續運行。
其中wait方法有三個over load方法:
wait()
wait(long)
wait(long,int)
wait方法通過參數可以指定等待的時長。如果沒有指定參數,默認一直等待直到被通知。
以下是一個演示代碼,以最簡潔的方式說明復雜的問題:
簡要說明下:
NotifyThread是用來模擬3秒鍾后通知其他等待狀態的線程的線程類;
WaitThread是用來模擬等待的線程類;
等待的中間對象是flag,一個String對象;
main方法中同時啟動一個Notify線程和三個wait線程;
package com.dxz.synchronizeddemo; public class NotifyTest { private String flag = "true"; class NotifyThread extends Thread { public NotifyThread(String name) { super(name); } public void run() { try { sleep(3000);// 推遲3秒鍾通知 } catch (InterruptedException e) { e.printStackTrace(); } flag = "false"; flag.notify(); } }; class WaitThread extends Thread { public WaitThread(String name) { super(name); } public void run() { System.out.println(getName() + " flag:" + flag); while (!flag.equals("false")) { System.out.println(getName() + " begin waiting!"); long waitTime = System.currentTimeMillis(); try { flag.wait(); } catch (InterruptedException e) { e.printStackTrace(); } waitTime = System.currentTimeMillis() - waitTime; System.out.println("wait time :" + waitTime); } System.out.println(getName() + " end waiting!"); } } public static void main(String[] args) throws InterruptedException { System.out.println("Main Thread Run!"); NotifyTest test = new NotifyTest(); NotifyThread notifyThread = test.new NotifyThread("notify01"); WaitThread waitThread01 = test.new WaitThread("waiter01"); WaitThread waitThread02 = test.new WaitThread("waiter02"); WaitThread waitThread03 = test.new WaitThread("waiter03"); notifyThread.start(); waitThread01.start(); waitThread02.start(); waitThread03.start(); } }
結果:
Main Thread Run! Exception in thread "waiter03" waiter03 flag:true waiter02 flag:true waiter03 begin waiting! waiter01 flag:true waiter02 begin waiting! waiter01 begin waiting! Exception in thread "waiter02" Exception in thread "waiter01" java.lang.IllegalMonitorStateException at java.lang.Object.wait(Native Method) at java.lang.Object.wait(Object.java:502) at com.dxz.synchronizeddemo.NotifyTest$WaitThread.run(NotifyTest.java:34) java.lang.IllegalMonitorStateException at java.lang.Object.wait(Native Method) at java.lang.Object.wait(Object.java:502) at com.dxz.synchronizeddemo.NotifyTest$WaitThread.run(NotifyTest.java:34) java.lang.IllegalMonitorStateException at java.lang.Object.wait(Native Method) at java.lang.Object.wait(Object.java:502) at com.dxz.synchronizeddemo.NotifyTest$WaitThread.run(NotifyTest.java:34) Exception in thread "notify01" java.lang.IllegalMonitorStateException at java.lang.Object.notify(Native Method) at com.dxz.synchronizeddemo.NotifyTest$NotifyThread.run(NotifyTest.java:19)
在wait和notify的地方都報錯java.lang.IllegalMonitorStateException,前面也講過,wait和notify方法一定要在synchronized里面,更具體點說有:
- 任何一個時刻,對象的控制權(monitor)只能被一個線程擁有。
- 無論是執行對象的wait、notify還是notifyAll方法,必須保證當前運行的線程取得了該對象的控制權(monitor)
- 如果在沒有控制權的線程里執行對象的以上三種方法,就會報java.lang.IllegalMonitorStateException異常。
- JVM基於多線程,默認情況下不能保證運行時線程的時序性
基於以上幾點事實,我們需要確保讓線程擁有對象的控制權。
也就是說在waitThread中執行wait方法時,要保證waitThread對flag有控制權;
在notifyThread中執行notify方法時,要保證notifyThread對flag有控制權。
線程取得控制權的方法有三:見《Synchronized之一:基本使用》
- 同步的實例方法(鎖用的是其實例對象本身。所有的非靜態同步方法執行需要順序執行,即不能並行執行。)
- 同步的靜態方法(鎖用的是其類對象本身。所有的靜態同步方法執行需要順序執行,即不能並行執行。)
- 實例方法中的同步塊(鎖是自己指定的,但不能是引用性對象及null對象)
- 靜態方法中的同步塊(鎖是自己指定的,但不能是引用性對象及null對象)
//nofity放入synchronized中 synchronized (flag) { flag = "false"; flag.notify(); } //wait放入synchronized中 synchronized (flag) { while (!flag.equals("false")) { System.out.println(getName() + " begin waiting!"); long waitTime = System.currentTimeMillis(); try { flag.wait(); } catch (InterruptedException e) { e.printStackTrace(); } waitTime = System.currentTimeMillis() - waitTime; System.out.println("wait time :" + waitTime); } }
再運行的結果:
Main Thread Run!
waiter02 flag:true
waiter01 flag:true
waiter03 flag:true
waiter02 begin waiting!
waiter03 begin waiting!
waiter01 begin waiting!
Exception in thread "notify01" java.lang.IllegalMonitorStateException
at java.lang.Object.notify(Native Method)
at com.dxz.synchronizeddemo.NotifyTest$NotifyThread.run(NotifyTest.java:20)
在flag.notify();的地方還是會報錯java.lang.IllegalMonitorStateException。這時的異常是由於在針對flag對象同步塊中,更改了flag對象的狀態所導致的。如下:
flag="false"; //改變的對象的內容 flag.notify();
對在同步塊中對flag進行了賦值操作,使得flag引用的對象改變,這時候再調用notify方法時,因為沒有控制權所以拋出異常。
我們可以改進一下,將flag改成一個JavaBean,然后更改它的屬性不會影響到flag的引用。
我們這里改成數組來試試,也可以達到同樣的效果:
這時候再運行,不再報異常,但是線程沒有結束是吧,沒錯,還有線程堵塞,處於wait狀態。
原因很簡單,我們有三個wait線程,只有一個notify線程,notify線程運行notify方法的時候,是隨機通知一個正在等待的線程,所以,現在應該還有兩個線程在waiting。
我們只需要將NotifyThread線程類中的flag.notify()方法改成notifyAll()就可以了。notifyAll方法會通知所有正在等待對象控制權的線程。
最終完成版如下:
package com.dxz.synchronizeddemo; public class NotifyTest2 { private String flag[] = {"true"}; class NotifyThread extends Thread { public NotifyThread(String name) { super(name); } public void run() { try { sleep(3000);// 推遲3秒鍾通知 } catch (InterruptedException e) { e.printStackTrace(); } synchronized (flag) { flag[0] = "false"; flag.notify(); } } }; class WaitThread extends Thread { public WaitThread(String name) { super(name); } public void run() { synchronized (flag) { System.out.println(getName() + " flag:" + flag); while (!flag[0].equals("false")) { System.out.println(getName() + " begin waiting!"); long waitTime = System.currentTimeMillis(); try { flag.wait(); } catch (InterruptedException e) { e.printStackTrace(); } waitTime = System.currentTimeMillis() - waitTime; System.out.println("wait time :" + waitTime); } System.out.println(getName() + " end waiting!"); } } } public static void main(String[] args) throws InterruptedException { System.out.println("Main Thread Run!"); NotifyTest2 test = new NotifyTest2(); NotifyThread notifyThread = test.new NotifyThread("notify01"); WaitThread waitThread01 = test.new WaitThread("waiter01"); WaitThread waitThread02 = test.new WaitThread("waiter02"); WaitThread waitThread03 = test.new WaitThread("waiter03"); notifyThread.start(); waitThread01.start(); waitThread02.start(); waitThread03.start(); } }
notify()和notifyAll()的本質區別
notify()和notifyAll()都是Object對象用於通知處在等待該對象的線程的方法。兩者的最大區別在於:
notifyAll使所有原來在該對象上等待被notify的所有線程統統退出wait的狀態,變成等待該對象上的鎖,一旦該對象被解鎖,他們就會去競爭。
notify則文明得多,它只是選擇一個wait狀態線程進行通知,並使它獲得該對象上的鎖,但不驚動其他同樣在等待被該對象notify的線程們,當第一個線程運行完畢以后釋放對象上的鎖此時如果該對象沒有再次使用notify語句,則即便該對象已經空閑,其他wait狀態等待的線程由於沒有得到該對象的通知,繼續處在wait狀態,直到這個對象發出一個notify或notifyAll,它們等待的是被notify或notifyAll,而不是鎖。
下面是一個很好的例子:
package com.dxz.synchronizeddemo; public class NotifyTest3 { private String flag[] = { "true" }; class NotifyThread extends Thread { public NotifyThread(String name) { super(name); } public void run() { try { sleep(3000);// 推遲3秒鍾通知 } catch (InterruptedException e) { e.printStackTrace(); } synchronized (flag) { flag[0] = "false"; flag.notifyAll(); } } }; class WaitThread extends Thread { public WaitThread(String name) { super(name); } public void run() { System.out.println(getName() + " flag:" + flag); synchronized (flag) { System.out.println(getName() + " flag:" + flag); while (!flag[0].equals("false")) { System.out.println(getName() + " begin waiting!"); long waitTime = System.currentTimeMillis(); try { flag.wait(); } catch (InterruptedException e) { e.printStackTrace(); } waitTime = System.currentTimeMillis() - waitTime; System.out.println("wait time :" + waitTime); } System.out.println(getName() + " end waiting!"); } System.out.println(getName() + " end waiting!"); } } public static void main(String[] args) throws InterruptedException { System.out.println("Main Thread Run!"); NotifyTest3 test = new NotifyTest3(); NotifyThread notifyThread = test.new NotifyThread("notify01"); WaitThread waitThread01 = test.new WaitThread("waiter01"); WaitThread waitThread02 = test.new WaitThread("waiter02"); WaitThread waitThread03 = test.new WaitThread("waiter03"); notifyThread.start(); waitThread01.start(); waitThread02.start(); waitThread03.start(); } }
結果:
Main Thread Run! waiter01 flag:[Ljava.lang.String;@584aceca waiter01 flag:[Ljava.lang.String;@584aceca waiter02 flag:[Ljava.lang.String;@584aceca waiter01 begin waiting! waiter02 flag:[Ljava.lang.String;@584aceca waiter02 begin waiting! waiter03 flag:[Ljava.lang.String;@584aceca waiter03 flag:[Ljava.lang.String;@584aceca waiter03 begin waiting! wait time :3000 waiter03 end waiting! waiter03 end waiting! wait time :3001 waiter02 end waiting! waiter02 end waiting! wait time :3001 waiter01 end waiting! waiter01 end waiting!