所謂回調(diào),定義是“一個(gè)方法的指針傳遞給事件源,當(dāng)某一事件發(fā)生時(shí)用來(lái)調(diào)用這個(gè)方法。”
比如客戶程序C調(diào)用服務(wù)程序S中的某個(gè)函數(shù)A,然后S又在某個(gè)時(shí)候反過(guò)來(lái)調(diào)用C中的某個(gè)函數(shù)B,對(duì)于C來(lái)說(shuō),這個(gè)B便叫做回調(diào)函數(shù)。例如Win32下的窗口過(guò)程函數(shù)就是一個(gè)典型的回調(diào)函數(shù)。一般說(shuō)來(lái),C不會(huì)自己調(diào)用B,C提供B的目的就是讓S來(lái)調(diào)用它,而且是C不得不提供。由于S并不知道C提供的B姓甚名誰(shuí),所以S會(huì)約定B的接口規(guī)范(函數(shù)原型),然后由C提前通過(guò)S的一個(gè)函數(shù)A告訴S自己將要使用B函數(shù),這個(gè)過(guò)程稱為回調(diào)函數(shù)的注冊(cè),A稱為注冊(cè)函數(shù)。
總的來(lái)說(shuō),回調(diào)函數(shù)就是那些自己寫的,但是不是自己來(lái)調(diào),而是給別人來(lái)調(diào)的函數(shù)。
回調(diào)一般用于層間協(xié)作,上層將本層函數(shù)安裝在下層,這個(gè)函數(shù)就是回調(diào)函數(shù),而下層在一定條件下觸發(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ù)指針。
消息響應(yīng)函數(shù)就可以看成是回調(diào)函數(shù),因?yàn)槭亲屜到y(tǒng)在合適的時(shí)候去調(diào)用。只不過(guò)消息響應(yīng)函數(shù)就是為了處理消息的,所以就拿出來(lái)單做一類了。其實(shí)本質(zhì)上就是回調(diào)函數(shù)。但是回調(diào)函數(shù)不是只有消息響應(yīng)函數(shù)一種,比如在內(nèi)核編程中,驅(qū)動(dòng)程序就要提供一些回調(diào)函數(shù),當(dāng)一個(gè)設(shè)備的數(shù)據(jù)讀寫完成后,讓系統(tǒng)調(diào)用這些回調(diào)函數(shù)來(lái)執(zhí)行一些后續(xù)工作。
其實(shí):回調(diào)就是該函數(shù)寫在高層,低層通過(guò)一個(gè)函數(shù)指針保存這個(gè)函數(shù),在某個(gè)事件的觸發(fā)下,低層通過(guò)該函數(shù)指針調(diào)用高層那個(gè)函數(shù)
C語(yǔ)言中回調(diào)函數(shù)解釋:
我們調(diào)用函數(shù)的方法有兩種:
直接調(diào)用:在函數(shù)A的函數(shù)體里通過(guò)書(shū)寫函數(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í)行。
比起直接調(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ù),被調(diào)函數(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è)通俗的例子:
某天,我打電話向你請(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ī)范。
java回調(diào)機(jī)制:
軟件模塊之間總是存在著一定的接口,從調(diào)用方式上,可以把他們分為三類:同步調(diào)用、回調(diào)和異步調(diào)用。
同步調(diào)用:一種阻塞式調(diào)用,調(diào)用方要等待對(duì)方執(zhí)行完畢才返回,它是一種單向調(diào)用;
回調(diào):一種雙向調(diào)用模式,也就是說(shuō),被調(diào)用方在接口被調(diào)用時(shí)也會(huì)調(diào)用對(duì)方的接口;
異步調(diào)用:一種類似消息或事件的機(jī)制,不過(guò)它的調(diào)用方向剛好相反,接口的服務(wù)在收到某種訊息或發(fā)生某種事件時(shí),會(huì)主動(dòng)通知客戶方(即調(diào)用客戶方的接口)。
回調(diào)和異步調(diào)用的關(guān)系非常緊密:使用回調(diào)來(lái)實(shí)現(xiàn)異步消息的注冊(cè),通過(guò)異步調(diào)用來(lái)實(shí)現(xiàn)消息的通知。
用Java里的例子:
public interface ICallBack//接口,規(guī)范需要有的函數(shù)
{
public void postExec();//需要回調(diào)的方法
}
另外的一個(gè)類:
public class FooBar
{ private ICallBackcallBack;
public voidsetCallBack(ICallBack callBack)
{
this.callBack = callBack;
doSth();
}
public voiddoSth()
{
callBack.postExec(); //調(diào)用回調(diào)函數(shù)
}
}
第二個(gè)類在測(cè)試類里面,是一個(gè)匿名類:
public class Test
{
public static void main(String[] args)
{
FooBar foo = new FooBar();
foo.setCallBack(new ICallBack() {
public void postExec()
{
System.out.println("在Test類中實(shí)現(xiàn)但由FooBar對(duì)象調(diào)用");
}實(shí)現(xiàn)了IcallBack接口的匿名類。
});
}
}
上訴的代碼:
1.兩個(gè)類:匿名類和FooBar
2.匿名類實(shí)現(xiàn)接口ICallBack(在test測(cè)試的main方法中用匿名類的形式實(shí)現(xiàn))
3.FooBar擁有一個(gè)參數(shù)為ICallBack接口類型的函數(shù)setCallBack(ICallBack)
4.匿名類運(yùn)行時(shí)調(diào)用FooBar中setCallBack函數(shù),以自身傳入?yún)?shù)
5.FooBar已取得匿名類,就可以隨時(shí)回調(diào)匿名類中所實(shí)現(xiàn)的ICallBack接口中的方法
假設(shè)有接口名為ICallBack 其中有方法名為postExec()。有類Myclass 實(shí)現(xiàn)了該接口,也就是一定實(shí)現(xiàn)了postExec()這個(gè)方法?,F(xiàn)在有另一個(gè)類FooBar它有個(gè)方法setCallBack(ICallBack callBack),并且setCallBack方法調(diào)用了callBack的postExec()方法。
如果現(xiàn)在,我們使用一個(gè)Myclass 的實(shí)例myClass,將它作為參數(shù)帶入到setCallBack(ICallBackcallBack)方法中,我們就說(shuō)setCallBack(ICallBackcallBack)方法回調(diào)了myClass的postExec()方法。
愛(ài)華網(wǎng)本文地址 » http://www.klfzs.com/a/25101014/229889.html
愛(ài)華網(wǎng)



