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ù)名都寫進(jìn)函數(shù)體了,經(jīng)過編譯器那么一編譯,板上釘釘,A注定調(diào)用的是B了,這樣的程序只能按照程序員事先設(shè)計(jì)好的流程執(zhí)行下去,太呆板了。此時(shí),間接調(diào)用的巨大靈活性就顯現(xiàn)出來了。想一想,如果p是函數(shù)A的一個(gè)參數(shù)(參數(shù)是變量,是變量就可以變嗎?。?,那么程序的最終用戶完全可以通過操作來改變p的指向——這樣,A在通過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)的接口來編寫程序,使它調(diào)用,這個(gè)就稱為回調(diào)。在調(diào)用接口時(shí),需要嚴(yán)格的按照定義的參數(shù)和方法調(diào)用,并且需要處理函數(shù)的異步,否則會(huì)導(dǎo)致程序的崩潰。這樣的解釋似乎還是比較難懂,這里舉個(gè)簡單的例子,程序員A寫了一段程序(程序a),其中預(yù)留有回調(diào)函數(shù)接口,并封裝好了該程序。程序員B要讓a調(diào)用自己的程序b中的一個(gè)方法,于是,他通過a中的接口回調(diào)自己b中的方法。目的達(dá)到。在C/C++中,要用回調(diào)函數(shù),被掉函數(shù)需要告訴調(diào)用者自己的指針地址,但在JAVA中沒有指針,怎么辦?我們可以通過接口(interface)來實(shí)現(xiàn)定義回調(diào)函數(shù)。
正常情況下開發(fā)人員使用已經(jīng)定義好的API,這個(gè)過程叫Call。但是有時(shí)這樣不能滿足需求,就需要程序員注冊自己的程序,然后讓事先定義好多API在合適的時(shí)候調(diào)用注冊的方法,這叫CallBack。
“通常大家說的回調(diào)函數(shù)一般就是按照別人(李四)的定好的接口規(guī)范寫,等待別人(張三)調(diào)用的函數(shù),在C語言中,回調(diào)函數(shù)通常通過函數(shù)指針來傳遞;在Java中,通常就是編寫另外一個(gè)類或類庫的人(李四)規(guī)定一個(gè)接口,然后你(張三)來實(shí)現(xiàn)這個(gè)接口,然后把這個(gè)實(shí)現(xiàn)類的一個(gè)對象作為參數(shù)傳給別人的程序,別人的程序必要時(shí)就會(huì)通過那個(gè)接口來調(diào)用你編寫的函數(shù)?!?/p>
使用技巧:定一個(gè)接口,在接口中聲明我們想調(diào)用的方法。
在另一個(gè)方法中注冊剛定義的回調(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),和接口中方法的返回值無關(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í)候反過來調(diào)用CLIENT中的某個(gè)函數(shù)CB,對于CLIENT來說,這個(gè)CB便叫做回調(diào)函數(shù)。例如Win32下的窗口過程函數(shù)就是一個(gè)典型的回調(diào)函數(shù)。
一般說來,CLIENT不會(huì)自己調(diào)用CB,CLIENT提供CB的目的就是讓SERVER來調(diào)用它,而且是CLIENT不得不提供。由于SERVER并不知道CLIENT提供的CB叫甚名誰,所以SERVER會(huì)約定CB的接口規(guī)范(函數(shù)原型),然后由CLIENT提前通過SERVER的一個(gè)函數(shù)R告訴SERVER自己將要使用CB函數(shù),這個(gè)過程稱為回調(diào)函數(shù)的注冊,R稱為注冊函數(shù)。Web SERVERerviCliente以及Java的RMI都用到回調(diào)機(jī)制,可以訪問遠(yuǎn)程服務(wù)器程序。
下面舉個(gè)通俗的例子:
某天,我打電話向你請教問題,當(dāng)然是個(gè)難題,:),你一時(shí)想不出解決方法,我又不能拿著電話在那里傻等,于是我們約定:等你想出辦法后打手機(jī)通知我,這樣,我就掛掉電話辦其它事情去了。過了XX分鐘,我的手機(jī)響了,你興高采烈的說問題已經(jīng)搞定,應(yīng)該如此這般處理。故事到此結(jié)束。這個(gè)例子說明了“異步+回調(diào)”的編程模式。其中,你后來打手機(jī)告訴我結(jié)果便是一個(gè)“回調(diào)”過程;我的手機(jī)號碼必須在以前告訴你,這便是注冊回調(diào)函數(shù);我的手機(jī)號碼應(yīng)該有效并且手機(jī)能夠接收到你的呼叫,這是回調(diào)函數(shù)必須符合接口規(guī)范。
通過上面?zhèn)€人感覺到回調(diào)函數(shù)更多的應(yīng)用就是結(jié)合異步。比如:ajax中jServer通過組件和服務(wù)器的異步通信。
什么是回調(diào)函數(shù)
精妙比喻:回調(diào)函數(shù)還真有點(diǎn)像您隨身帶的BP機(jī):告訴別人號碼,在它有事情時(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)用層來做進(jìn)一步處理,這在分層的數(shù)據(jù)通信中很普遍。
其實(shí)回調(diào)和API非常接近,他們的共性都是跨層調(diào)用的函數(shù)。但區(qū)別是API是低層提供給高層的調(diào)用,一般這個(gè)函數(shù)對高層都是已知的;而回調(diào)正好相反,他是高層提供給底層的調(diào)用,對于低層他是未知的,必須由高層進(jìn)行安裝,這個(gè)安裝函數(shù)其實(shí)就是一個(gè)低層提供的API,安裝后低層不知道這個(gè)回調(diào)的名字,但它通過一個(gè)函數(shù)指針來保存這個(gè)回調(diào),在需要調(diào)用時(shí),只需引用這個(gè)函數(shù)指針和相關(guān)的參數(shù)指針。
其實(shí):回調(diào)就是該函數(shù)寫在高層,低層通過一個(gè)函數(shù)指針保存這個(gè)函數(shù),在某個(gè)事件的觸發(fā)下,低層通過該函數(shù)指針調(diào)用高層那個(gè)函數(shù)。
感謝閱讀,希望能幫助到大家,謝謝大家對本站的支持!
上一篇:深入剖析Java編程中的序列化
欄 目:Java編程
下一篇:舉例講解設(shè)計(jì)模式中的訪問者模式在Java編程中的運(yùn)用
本文標(biāo)題:Java 回調(diào)函數(shù)詳解及使用
本文地址:http://mengdiqiu.com.cn/a1/Javabiancheng/8473.html
您可能感興趣的文章
- 01-10Java咖啡館(1)——嘆咖啡
- 01-10Java Socket編程(三) 服務(wù)器Sockets
- 01-10Java進(jìn)階:Struts多模塊的技巧
- 01-10Java Socket編程(一) Socket傳輸模式
- 01-10Java Socket編程(二) Java面向連接的類
- 01-10Java運(yùn)行時(shí)多態(tài)性的實(shí)現(xiàn)
- 01-10Java經(jīng)驗(yàn)點(diǎn)滴:處理沒有被捕獲的異常
- 01-10Java Socket編程(四) 重復(fù)和并發(fā)服務(wù)器
- 01-10Java中的浮點(diǎn)數(shù)分析
- 01-10面向?qū)ο缶幊?Java中的抽象數(shù)據(jù)類型


閱讀排行
本欄相關(guān)
- 01-10Java咖啡館(1)——嘆咖啡
- 01-10JVM的垃圾回收機(jī)制詳解和調(diào)優(yōu)
- 01-10Java Socket編程(三) 服務(wù)器Sockets
- 01-10Java進(jìn)階:Struts多模塊的技巧
- 01-10J2SE 1.5版本的新特性一覽
- 01-10Java Socket編程(一) Socket傳輸模式
- 01-10Java運(yùn)行時(shí)多態(tài)性的實(shí)現(xiàn)
- 01-10Java Socket編程(二) Java面向連接的類
- 01-10Java Socket編程(四) 重復(fù)和并發(fā)服務(wù)
- 01-10Java經(jīng)驗(yàn)點(diǎn)滴:處理沒有被捕獲的異常
隨機(jī)閱讀
- 08-05dedecms(織夢)副欄目數(shù)量限制代碼修改
- 08-05織夢dedecms什么時(shí)候用欄目交叉功能?
- 01-10SublimeText編譯C開發(fā)環(huán)境設(shè)置
- 04-02jquery與jsp,用jquery
- 01-11Mac OSX 打開原生自帶讀寫NTFS功能(圖文
- 08-05DEDE織夢data目錄下的sessions文件夾有什
- 01-10C#中split用法實(shí)例總結(jié)
- 01-10delphi制作wav文件的方法
- 01-10使用C語言求解撲克牌的順子及n個(gè)骰子
- 01-11ajax實(shí)現(xiàn)頁面的局部加載