由多線程runable接口聯想的回調函數

runable 接口中的run方法就是回調函數。html


所謂回調,就是客戶程序C調用服務程序S中的某個函數A,而後S又在某個時候反過來調用C中的某個函數B,對於C來講,這個B便叫作回調函數。例如Win32下的窗口過程函數就是一個典型的回調函數。通常說來,C不會本身調用B,C提供B的目的就是讓S來調用它,並且是C不得不提供。因爲S並不知道C提供的B姓甚名誰,因此S會約定B的接口規範(函數原型),而後由C提早經過S的一個函數R告訴S本身將要使用B函數,這個過程稱爲回調函數的註冊,R稱爲註冊函數。Web Service以及Java的RMI都用到回調機制,能夠訪問遠程服務器程序。java


    下面舉個通俗的例子:
    某天,我打電話向你請教問題,固然是個難題,^_^,你一時想不出解決方法,我又不能拿着電話在那裏傻等,因而咱們約定:等你想出辦法後打手機通知我,這樣,我就掛掉電話辦其它事情去了。過了XX分鐘,個人手機響了,你興高采烈的說問題已經搞定,應該如此這般處理。故事到此結束。這個例子說明了「異步+回調」的編程模式。其中,你後來打手機告訴我結果即是一個「回調」過程;個人手機號碼必須在之前告訴你,這即是註冊回調函數;個人手機號碼應該有效而且手機可以接收到你的呼叫,這是回調函數必須符合接口規範。
程序員


    經過上面我的感受到回調更多的應用就是結合異步。好比:Ajax中js經過組件和服務器的異步通訊。編程

 

例:服務器

    程序員A寫了一段程序(程序a),其中預留有回調函數接口,並封裝好了該程序。程序員B要讓a調用本身的程序b中的一個方法,因而,他經過a中的接口回調本身b中的方法。目的達到。在C/C++中,要用回調函數,被掉函數須要告訴調用者本身的指針地址,但在JAVA中沒有指針,怎麼辦?咱們能夠經過接口(interface)來實現定義回調函數。
     假設我是程序員A,如下是個人程序a:

多線程

[java]  view plain copy print ?
  1. public class Caller  
  2. {  
  3.     public MyCallInterface mc;  
  4.   
  5.     public void setCallfuc(MyCallInterface mc)  
  6.     {  
  7.        this.mc= mc;  
  8.     }  
  9.   
  10.     public void call(){  
  11.        this.mc.method();  
  12.     }  
  13. }      
  

 

     我還須要定義一個接口,以便程序員B根據個人定義編寫程序實現接口。
app

[java]  view plain copy print ?
  1. public interface MyCallInterface  
  2. {  
  3.     public void method();  
  4.   
  5. }  

     因而,程序員B只須要實現這個接口就能達到回調的目的了:
異步

[java]  view plain copy print ?
  1. public class B implements MyCallInterface  
  2. {  
  3.     public void method()  
  4.     {  
  5.        System.out.println("回調");  
  6.     }  
  7.   
  8.     public static void main(String args[])  
  9.     {  
  10.        Caller call = new Caller();  
  11.        call.setCallfuc(new B());  
  12.        call.call();  
  13.     }  

  1. }  


又一個詳細的例子:

詳細的實例以下:函數

一、建立一個回調接口:this

1  // 回調接口
2    public   interface  ICallBack
3 
4  void  run();
5  }

二、建立回調接口的實現類:

1  class  CallBackClass  implements  ICallBack
2  { public   void  run()
3 
4    // 輸出當前時間
5     System.out.println(System.currentTimeMillis() );
6  }
7  }

三、建立控制類

1  class  Controller
2  {
3  public  ICallBack CallBackObject  =   null ; //  引用回調對象
4  Scanner input  =   new  Scanner(System.in);  // 讀取命令行輸入
5  public  Controller(ICallBack obj)
6  {
7  this .CallBackObject  =  obj;
8  }
9  public   void  Begin()
10  {
11  while (input.next()  !=   null ) // 判斷是否有輸入
12  {
13  CallBackObject.run();
14  }
15  }
16  }

運行程序:

1  class  Program
2  {
3  static   void  Main(string[] args)
4  {
5  // 建立控制器對象,將提供給它的回調對象傳入
6  Controller obj  =   new  Controller( new  CallBackClass());
7  // 啓動控制器對象運行
8  obj.Begin();
9  }
10  }

複製代碼

 

 

總結以下:

實現多線程中用到的實現runable接口重寫run方法,本質是回調函類的實現, run方法也是重寫的回調函數的方法。

從run開始,依次的調用邏輯爲:

1. 重寫的run方法被執行,

2. run是接口的實現類裏頭的方法。

3. 而接口的實現類爲回調控制器服務的,在回調控制器中被實例化的,實例化以後,變量就有值了,而run方法每次被調用代碼都被執行一遍。

4. 主函數裏頭調用的回調控制器。

相關文章
相關標籤/搜索