Java 回調(diào)函數(shù)詳解及使用
Java 回調(diào)函數(shù)詳解
前言:
C語言中回調(diào)函數(shù)解釋:
回調(diào)函數(shù)(Callback Function)是怎樣一種函數(shù)呢?
函數(shù)是用來被調(diào)用的,我們調(diào)用函數(shù)的方法有兩種:
直接調(diào)用:在函數(shù)A的函數(shù)體里通過書寫函數(shù)B的函數(shù)名來調(diào)用之,使內(nèi)存中對應(yīng)函數(shù)B的代碼得以執(zhí)行。這里,A稱為“主叫函數(shù)”(Caller),B稱為“被叫函數(shù)”(Callee)。
間接調(diào)用:在函數(shù)A的函數(shù)體里并不出現(xiàn)函數(shù)B的函數(shù)名,而是使用指向函數(shù)B的函數(shù)指針p來使內(nèi)存中屬于函數(shù)B的代碼片斷得以執(zhí)行——聽起來很酷,是吧。
比起直接調(diào)用來,間接調(diào)用的確麻煩,那為什么還要使用間接調(diào)用呢?原因很簡單——直接調(diào)用把函數(shù)名都寫進函數(shù)體了,經(jīng)過編譯器那么一編譯,板上釘釘,A注定調(diào)用的是B了,這樣的程序只能按照程序員事先設(shè)計好的流程執(zhí)行下去,太呆板了。此時,間接調(diào)用的巨大靈活性就顯現(xiàn)出來了。想一想,如果p是函數(shù)A的一個參數(shù)(參數(shù)是變量,是變量就可以變嗎?。?,那么程序的最終用戶完全可以通過操作來改變p的指向——這樣,A在通過p調(diào)用函數(shù)的時候就有機會調(diào)用到不同的函數(shù),這樣程序的實用性和擴展性就強多了。
在WINDOWS中,程序員想讓系統(tǒng)DLL調(diào)用自己編寫的一個方法,于是利用DLL當中回調(diào)函數(shù)(CALLBACK)的接口來編寫程序,使它調(diào)用,這個就稱為回調(diào)。在調(diào)用接口時,需要嚴格的按照定義的參數(shù)和方法調(diào)用,并且需要處理函數(shù)的異步,否則會導(dǎo)致程序的崩潰。這樣的解釋似乎還是比較難懂,這里舉個簡單的例子,程序員A寫了一段程序(程序a),其中預(yù)留有回調(diào)函數(shù)接口,并封裝好了該程序。程序員B要讓a調(diào)用自己的程序b中的一個方法,于是,他通過a中的接口回調(diào)自己b中的方法。目的達到。在C/C++中,要用回調(diào)函數(shù),被掉函數(shù)需要告訴調(diào)用者自己的指針地址,但在JAVA中沒有指針,怎么辦?我們可以通過接口(interface)來實現(xiàn)定義回調(diào)函數(shù)。
正常情況下開發(fā)人員使用已經(jīng)定義好的API,這個過程叫Call。但是有時這樣不能滿足需求,就需要程序員注冊自己的程序,然后讓事先定義好多API在合適的時候調(diào)用注冊的方法,這叫CallBack。
“通常大家說的回調(diào)函數(shù)一般就是按照別人(李四)的定好的接口規(guī)范寫,等待別人(張三)調(diào)用的函數(shù),在C語言中,回調(diào)函數(shù)通常通過函數(shù)指針來傳遞;在Java中,通常就是編寫另外一個類或類庫的人(李四)規(guī)定一個接口,然后你(張三)來實現(xiàn)這個接口,然后把這個實現(xiàn)類的一個對象作為參數(shù)傳給別人的程序,別人的程序必要時就會通過那個接口來調(diào)用你編寫的函數(shù)?!?/p>
使用技巧:定一個接口,在接口中聲明我們想調(diào)用的方法。
在另一個方法中注冊剛定義的回調(diào)接口
第一步:定義回調(diào)接口
package com.callback; /** * @since 2012-02-02 * @desc 定義回調(diào)接口 */ public interface CallBackInterface { public void doSome(); public void exectueMethod(); }
第二步:李四編寫的程序
package com.callback; /** * @since 2012-02-02 * @desc 李四 */ public class MethodB { public double getTime(CallBackInterface callBack) { long start = System.currentTimeMillis(); callBack.exectueMethod(); long end = System.currentTimeMillis(); System.out.println("cost time=" + (end - start)); return end - start; } }
第三步:張三實現(xiàn)李四規(guī)定的接口
package com.callback; /** * @since 2012-02-02 * @desc 張三 */ public class MethodA { public static void main(String args[]){ MethodB b=new MethodB(); //返回值d只和MethodB有關(guān),和接口中方法的返回值無關(guān) double d=b.getTime(new CallBackInterface() { //張三實現(xiàn)了李四定義的接口 public void exectueMethod() { new MethodA().testMethod(); } @Override public void doSome() { } }); System.out.println("d="+d); } public void testMethod(){ for(int i=0;i<10000;i++){ System.out.print(""); } } }
輸出結(jié)果:
cost time=31 d=31.0
理解“回調(diào)函數(shù)”
所謂回調(diào),就是客戶程序CLIENT調(diào)用服務(wù)程序SERVER中的某個函數(shù)SA,然后SERVER又在某個時候反過來調(diào)用CLIENT中的某個函數(shù)CB,對于CLIENT來說,這個CB便叫做回調(diào)函數(shù)。例如Win32下的窗口過程函數(shù)就是一個典型的回調(diào)函數(shù)。
一般說來,CLIENT不會自己調(diào)用CB,CLIENT提供CB的目的就是讓SERVER來調(diào)用它,而且是CLIENT不得不提供。由于SERVER并不知道CLIENT提供的CB叫甚名誰,所以SERVER會約定CB的接口規(guī)范(函數(shù)原型),然后由CLIENT提前通過SERVER的一個函數(shù)R告訴SERVER自己將要使用CB函數(shù),這個過程稱為回調(diào)函數(shù)的注冊,R稱為注冊函數(shù)。Web SERVERerviCliente以及Java的RMI都用到回調(diào)機制,可以訪問遠程服務(wù)器程序。
下面舉個通俗的例子:
某天,我打電話向你請教問題,當然是個難題,:),你一時想不出解決方法,我又不能拿著電話在那里傻等,于是我們約定:等你想出辦法后打手機通知我,這樣,我就掛掉電話辦其它事情去了。過了XX分鐘,我的手機響了,你興高采烈的說問題已經(jīng)搞定,應(yīng)該如此這般處理。故事到此結(jié)束。這個例子說明了“異步+回調(diào)”的編程模式。其中,你后來打手機告訴我結(jié)果便是一個“回調(diào)”過程;我的手機號碼必須在以前告訴你,這便是注冊回調(diào)函數(shù);我的手機號碼應(yīng)該有效并且手機能夠接收到你的呼叫,這是回調(diào)函數(shù)必須符合接口規(guī)范。
通過上面?zhèn)€人感覺到回調(diào)函數(shù)更多的應(yīng)用就是結(jié)合異步。比如:ajax中jServer通過組件和服務(wù)器的異步通信。
什么是回調(diào)函數(shù)
精妙比喻:回調(diào)函數(shù)還真有點像您隨身帶的BP機:告訴別人號碼,在它有事情時Call您
回調(diào)用于層間協(xié)作,上層將本層函數(shù)安裝在下層,這個函數(shù)就是回調(diào),而下層在一定條件下觸發(fā)回調(diào),例如作為一個驅(qū)動,是一個底層,他在收到一個數(shù)據(jù)時,除了完成本層的處理工作外,還將進行回調(diào),將這個數(shù)據(jù)交給上層應(yīng)用層來做進一步處理,這在分層的數(shù)據(jù)通信中很普遍。
其實回調(diào)和API非常接近,他們的共性都是跨層調(diào)用的函數(shù)。但區(qū)別是API是低層提供給高層的調(diào)用,一般這個函數(shù)對高層都是已知的;而回調(diào)正好相反,他是高層提供給底層的調(diào)用,對于低層他是未知的,必須由高層進行安裝,這個安裝函數(shù)其實就是一個低層提供的API,安裝后低層不知道這個回調(diào)的名字,但它通過一個函數(shù)指針來保存這個回調(diào),在需要調(diào)用時,只需引用這個函數(shù)指針和相關(guān)的參數(shù)指針。
其實:回調(diào)就是該函數(shù)寫在高層,低層通過一個函數(shù)指針保存這個函數(shù),在某個事件的觸發(fā)下,低層通過該函數(shù)指針調(diào)用高層那個函數(shù)。
感謝閱讀,希望能幫助到大家,謝謝大家對本站的支持!
相關(guān)文章
java并發(fā)編程關(guān)鍵字volatile保證可見性不保證原子性詳解
這篇文章主要為大家介紹了java并發(fā)編程關(guān)鍵字volatile保證可見性不保證原子性詳解,文中附含詳細示例說明,有需要的朋友可以借鑒參考下,希望能夠有所幫助2022-02-02