再談java回調(diào)函數(shù)
又遇到了回調(diào)函數(shù),這次打算寫(xiě)下來(lái)分享一下。
所謂回調(diào)函數(shù),或者在面向?qū)ο笳Z(yǔ)言里叫回調(diào)方法,簡(jiǎn)單點(diǎn)講,就是回頭在某個(gè)時(shí)間(事件發(fā)生)被調(diào)用的函數(shù)。
再詳細(xì)點(diǎn):就是一個(gè)函數(shù)A,作為參數(shù),傳入了另一個(gè)函數(shù)B,然后被B在某個(gè)時(shí)間調(diào)用。
這里可以有疑問(wèn)了,既然是一個(gè)函數(shù)調(diào)用另一個(gè)函數(shù),可以在函數(shù)體里面調(diào)用啊,為什么還要把函數(shù)作為參數(shù)傳到另一個(gè)函數(shù)里被調(diào)用?何況還有一些語(yǔ)言(比如java)不支持把函數(shù)作為參數(shù)。
對(duì)的,確實(shí)可以在函數(shù)體里調(diào)用另一個(gè)函數(shù),功能上好像是沒(méi)差別的,但是這里有一個(gè)問(wèn)題,就是你要調(diào)用的這個(gè)函數(shù)被寫(xiě)死了,也就是說(shuō)這樣函數(shù)B只能調(diào)用函數(shù)A了,這樣如果在另一個(gè)情景下,有個(gè)與A不同實(shí)現(xiàn)的函數(shù)C也需要在B的某個(gè)時(shí)刻被調(diào)用,那怎么辦。
下面繼續(xù)說(shuō)回調(diào)函數(shù),在c/c++里,回調(diào)函數(shù)可以使用函數(shù)指針作為參數(shù)被另一個(gè)函數(shù)調(diào)用;在c#里,可以使用委托,如果是事件方法的話(huà),還有event關(guān)鍵字;在python和javascript里,可以直接把函數(shù)當(dāng)對(duì)象傳參,這些語(yǔ)言都很好實(shí)現(xiàn)回調(diào)函數(shù)(方法),可是, java呢? 先說(shuō)點(diǎn)題外話(huà),自從學(xué)了C#,就不喜歡java了,曾經(jīng)一度打算以后不再用java,可是現(xiàn)實(shí)并沒(méi)有那么理想,我現(xiàn)在要做android,所以還是不能放下java,而且今天遇到這個(gè)回調(diào)函數(shù)的問(wèn)題,也是從java里遇到的,我個(gè)人覺(jué)得,在這個(gè)博客里出現(xiàn)的語(yǔ)言,除了java外,對(duì)于回調(diào),都可以既容易,又好理解的實(shí)現(xiàn),但是java,我覺(jué)得并不是那樣,不然我也不會(huì)來(lái)寫(xiě)這篇博客。
好了繼續(xù)說(shuō),關(guān)于java中的回調(diào)方法的實(shí)現(xiàn)。這篇博客的重點(diǎn)就是說(shuō)java的。 在java中,回調(diào)方法是用借用接口來(lái)實(shí)現(xiàn)的,我在網(wǎng)上找到一句話(huà):
“把實(shí)現(xiàn)某一接口的類(lèi)所創(chuàng)建的對(duì)象的引用,賦值給該接口聲明的接口變量,那么該接口變量就可以調(diào)用被實(shí)現(xiàn)的接口的方法”。
很繞哈,簡(jiǎn)單解釋下:
有一個(gè)接口,接口里有一個(gè)方法(這個(gè)方法就是要回調(diào)的方法):
interface CallBackInterface { void callBackMethod(); }
我們知道,接口對(duì)象不能直接用,因?yàn)槔锩娴姆椒ǘ紱](méi)有實(shí)現(xiàn)。所以要找個(gè)類(lèi)實(shí)現(xiàn)這個(gè)接口。
所以現(xiàn)在加一個(gè)類(lèi),實(shí)現(xiàn)這個(gè)接口:
interface CallBackInterface { void callBackMethod(); } class CallBackClass implements CallBackInterface{ @Override public void callBackMethod() { System.out.println("hello"); } }
好了,最后一步:把實(shí)現(xiàn)了接口的類(lèi)的對(duì)象賦值給聲明的接口變量(我給寫(xiě)進(jìn)一個(gè)方法里了,然后外面加了個(gè)類(lèi)的殼子):
public class CallBackTest { interface CallBackInterface { void callBackMethod(); } class CallBackClass implements CallBackInterface { @Override public void callBackMethod() { System.out.println("hello"); } } public void showCallBack() { CallBackInterface itfs = new CallBackClass(); itfs.callBackMethod(); } }
現(xiàn)在可以調(diào)用試試看了:
public class Test { public static void main(String[] args) { new CallBackTest().showCallBack(); } }
沒(méi)意外的話(huà),會(huì)成功輸出hello,反正我這邊是的.
例子看完了,所以說(shuō)我做了什么呢? 再詳細(xì)點(diǎn)說(shuō),我們有一個(gè)要在某一個(gè)方法里被調(diào)用的方法(這個(gè)方法就是回調(diào)方法), 前面我們也說(shuō)了,最好不要直接把想要回調(diào)方法做的事直接寫(xiě)在調(diào)用方法里, 又因?yàn)閖ava里沒(méi)法把方法當(dāng)做參數(shù)傳遞,所以我們只好把這個(gè)回調(diào)方法放在了接口里(為什么不是類(lèi)?不是抽象類(lèi)?而是接口?你可以自己去找下抽象類(lèi)與接口的異同,自己解決這個(gè)問(wèn)題)。有接口的話(huà),就要被類(lèi)實(shí)現(xiàn),然后,只要是給接口的對(duì)象賦予實(shí)現(xiàn)類(lèi)的對(duì)象,這個(gè)接口的對(duì)象就可以調(diào)用那個(gè)方法了。理解這里的話(huà),有一個(gè)重點(diǎn),就是多態(tài), 這里用到的多態(tài)知識(shí)就是,接口的對(duì)象可以順利被子類(lèi)賦值,并且調(diào)用子類(lèi)的重寫(xiě)方法(類(lèi)也有類(lèi)似的概念)。
再多說(shuō)一點(diǎn),這里任何實(shí)現(xiàn)了CallbackInterface接口的類(lèi),都可以像下面這樣放在new后面(就是賦值):
public class CallBackTest { interface CallBackInterface { void callBackMethod(); } class CallBackClass implements CallBackInterface { @Override public void callBackMethod() { System.out.println("hello"); } } class Controller { private CallBackInterface cbitf; // 這個(gè)boolean只是為了模擬有事件,沒(méi)啥實(shí)用價(jià)值 public boolean somethingHappend; // 這里確實(shí)可以直接把CallBackClass做參數(shù),而且省掉接口的定義 // 但是這樣做的話(huà),就像是回調(diào)函數(shù)直接寫(xiě)在了調(diào)用函數(shù)里一樣 // 不明白的話(huà)就好好理解下"約定"和"調(diào)用者不管回調(diào)函數(shù)是怎么實(shí)現(xiàn)的"吧 public Controller(CallBackInterface itfs) { somethingHappend = true; this.cbitf = itfs; } public void doSomething() { if(somethingHappend) { cbitf.callBackMethod(); } } } public void showCallBack() { CallBackClass cbc = new CallBackClass(); Controller ctrlr = new Controller(cbc); ctrlr.doSomething(); // 其實(shí)上面也可以這樣寫(xiě)在一行里 // new Controller(new CallBackClass()).doSomething(); } }
最后多說(shuō)一點(diǎn),其實(shí)這種應(yīng)用在android里會(huì)經(jīng)常遇到,我就是在學(xué)android的時(shí)候遇到的。
以上就是個(gè)人對(duì)于回調(diào)函數(shù)的理解和使用方法了,希望大家能夠喜歡。
相關(guān)文章
JAVA-4NIO之Channel之間的數(shù)據(jù)傳輸方法
下面小編就為大家?guī)?lái)一篇JAVA-4NIO之Channel之間的數(shù)據(jù)傳輸方法。小編覺(jué)得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2017-06-06淺談圖片上傳利用request.getInputStream()獲取文件流時(shí)遇到的問(wèn)題
下面小編就為大家?guī)?lái)一篇淺談圖片上傳利用request.getInputStream()獲取文件流時(shí)遇到的問(wèn)題。小編覺(jué)得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2017-11-11Springboot中如何使用過(guò)濾器校驗(yàn)PSOT類(lèi)型請(qǐng)求參數(shù)內(nèi)容
在Springboot中創(chuàng)建過(guò)濾器,用來(lái)過(guò)濾所有POST類(lèi)型請(qǐng)求并獲取body中的參數(shù)進(jìn)行校驗(yàn)內(nèi)容是否合法,該方法僅適用于POST類(lèi)型請(qǐng)求,本文給大家介紹Springboot中如何使用過(guò)濾器校驗(yàn)PSOT類(lèi)型請(qǐng)求參數(shù)內(nèi)容,感興趣的朋友一起看看吧2023-08-08詳解Java中String,StringBuffer和StringBuilder的使用
這篇文章主要為大家詳細(xì)介紹了Java中String,StringBuffer和StringBuilder三者的區(qū)別以及使用,文中的少了講解詳細(xì),感興趣的可以了解一下2022-07-07Springboot獲取前端反饋信息并存入數(shù)據(jù)庫(kù)的實(shí)現(xiàn)代碼
這篇文章主要介紹了Springboot獲取前端反饋信息并存入數(shù)據(jù)庫(kù)的實(shí)現(xiàn)代碼,本文通過(guò)實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2021-03-03springboot+websocket實(shí)現(xiàn)并發(fā)搶紅包功能
本文主要介紹了springboot+websocket實(shí)現(xiàn)并發(fā)搶紅包功能,主要包含了4種步驟,文中通過(guò)示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2021-12-12