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