線程通信:
如果線程A和線程B持有同一個MyObject類的對象object,這兩個線程會去調用不同的方法,但是它們是同步執行的,比如:線程B需要等待線程A執行完了methodA()方法之後,它才能執行methodB()方法。這樣,線程A和線程B就實現了 通信。
線程通信中要用到的方法:wait()方法:
執行同步鎖(obj對象)的該方法的線程進入堵塞狀態,會釋放對象的鎖,java虛擬機把該線程放到該對象的等待池中,該線程如果要再次執行,則需要其他線程將它喚醒。
1 package tongxin; 2 3 public class MyThreadA extends Thread { 4 private Object obj; 5 public MyThreadA(String name,Object obj){ 6 7 super(name); 8 this.obj = obj; 9 10 } 11 public void run(){ 12 synchronized(obj){ //設置obj為同步鎖 13 14 for(int i = 0;i<10;i++){ 15 System.out.println(Thread.currentThread().getName()+"---"+i); 16 17 if(i == 5){ 18 try { 19 obj.wait(); 20 } catch (InterruptedException e) { 21 // TODO Auto-generated catch block 22 e.printStackTrace(); 23 } 24 try { 25 sleep(100); 26 } catch (InterruptedException e) { 27 // TODO Auto-generated catch block 28 e.printStackTrace(); 29 } 30 } 31 } 32 } 33 } 34 35 }
1 package tongxin; 2 3 public class Demo01 { 4 5 public static void main(String[] args) { 6 // TODO Auto-generated method stub 7 8 Object obj = new Object(); 9 MyThreadA mt1 = new MyThreadA("A",obj); 10 mt1.start(); 11 12 } 13 14 }
output:
1 A---0 2 A---1 3 A---2 4 A---3 5 A---4 6 A---5
當i = 5時,執行了同步鎖的wait()方法,然後線程A將同步鎖釋放進入堵塞狀態,如果A線程還有恢復運行,則需要另一個持相同同步鎖的線程將其喚醒。
喚醒線程的方法:notifi()
另一個線程在運行過程中執行了同步鎖的notifi方法, 會喚醒在對象的等待池中等待的一個線程(隨機選擇),java虛擬機會隨機選擇一個線程轉到對象的鎖池中。
注意一個線程要去喚醒另一個執行了wait()方法進入堵塞狀態的線程,則其要去執行同一個同步鎖的notifi()方法。
假如有一個房間,A進去之後把門鎖上,A在房間裡睡著了,B要把A叫醒,就要用鎖(同步鎖)上的鑰匙把門開開。A與B用的是同一把鎖,只不過A是在鎖門(wait()方法),B是在開門(notifi()方法)。
1 package tongxin; 2 3 public class MyThreadB extends Thread { 4 private Object obj; 5 public MyThreadB(String name,Object obj){ 6 7 super(name); 8 this.obj = obj; 9 10 } 11 public void run(){ 12 synchronized(obj){ //設置obj為同步鎖 13 14 for(int i = 0;i<10;i++){ 15 System.out.println(Thread.currentThread().getName()+"---"+i); 16 17 if(i == 2){ 18 obj.notify(); 19 try { 20 sleep(100); 21 } catch (InterruptedException e) { 22 // TODO Auto-generated catch block 23 e.printStackTrace(); 24 } 25 } 26 } 27 } 28 } 29 30 31 }
1 package tongxin; 2 3 public class Demo01 { 4 5 public static void main(String[] args) { 6 // TODO Auto-generated method stub 7 8 Object obj = new Object(); 9 MyThreadA mt1 = new MyThreadA("A",obj); 10 MyThreadB mt2 = new MyThreadB("B",obj); 11 mt1.start(); 12 mt2.start(); 13 14 15 } 16 17 }
output:
A---0 A---1 A---2 A---3 A---4 A---5 B---0 B---1 B---2 B---3 B---4 B---5 B---6 B---7 B---8 B---9 A---6 A---7 A---8 A---9
創建一個新的線程對象B來喚醒A線程,運行過程中,i = 2時,B喚醒A線程將其轉到對象的鎖池中,但此時由於B持有同步鎖所以B線程繼續運行,運行結束後釋放同步鎖,此時A獲取到同步鎖恢復運行。
notifyAll()方法:
該方法會喚醒在這個對象鎖池中的所有線程。
如有不對之處還望指正,謝謝。