Java回調(diào)方法詳解
回調(diào)在維基百科中定義為:
在計(jì)算機(jī)程序設(shè)計(jì)中,回調(diào)函數(shù),是指通過函數(shù)參數(shù)傳遞到其他代碼的,某一塊可執(zhí)行代碼的引用。
其目的是允許底層代碼調(diào)用在高層定義的子程序。
舉個(gè)例子可能更明白一些:以Android中用retrofit進(jìn)行網(wǎng)絡(luò)請(qǐng)求為例,這個(gè)是異步回調(diào)的一個(gè)例子。
在發(fā)起網(wǎng)絡(luò)請(qǐng)求之后,app可以繼續(xù)其他事情,網(wǎng)絡(luò)請(qǐng)求的結(jié)果一般是通過onResponse與onFailure這兩個(gè)方法返回得到??匆幌孪嚓P(guān)部分的代碼:
call.enqueue(new Callback<HistoryBean>() { @Override public void onResponse(Call<HistoryBean> call, Response<HistoryBean> response) { HistoryBean hb = response.body(); if(hb == null) return; showText.append(hb.isError() + ""); for(HistoryBean.ResultsBean rb : hb.getResults()){ showText.append(rb.getTitle() + "/n"); } } @Override public void onFailure(Call<HistoryBean> call, Throwable t) { } });
忽略上面CallBack中的泛型,按照維基百科中的定義,匿名內(nèi)部類里面的全部代碼可以看成函數(shù)參數(shù)傳遞到其他代碼的,某一塊可執(zhí)行代碼的引用。 onResponse與onFailure這兩個(gè)方法就是回調(diào)方法。底層的代碼就是已經(jīng)寫好不變的網(wǎng)絡(luò)請(qǐng)求部分,高層定義的子程序就是回調(diào),因?yàn)榫唧w的實(shí)現(xiàn)交給了使用者,所以具備了很高的靈活性。上面就是通過enqueue(Callback callback)這個(gè)方法來關(guān)聯(lián)起來的。
回調(diào)方法的步驟
上面說的回調(diào)是很通用的概念,放到程序書寫上面,就可以說:
A類中調(diào)用B類中的某個(gè)方法C,然后B類中在反過來調(diào)用A類中的方法D,在這里面D就是回調(diào)方法。B類就是底層的代碼,A類是高層的代碼。
所以通過上面的解釋,我們可以推斷出一些東西,為了表示D方法的通用性,我們采用接口的形式讓D方法稱為一個(gè)接口方法,那么如果B類要調(diào)用A類中的方法D,那勢(shì)必A類要實(shí)現(xiàn)這個(gè)接口,這樣,根據(jù)實(shí)現(xiàn)的不同,就會(huì)有多態(tài)性,使方法具備靈活性。
A類要調(diào)用B類中的某個(gè)方法C,那勢(shì)必A類中必須包含B的引用,要不然是無法調(diào)用的,這一步稱之為注冊(cè)回調(diào)接口。那么如何實(shí)現(xiàn)B類中反過來調(diào)用A類中的方法D呢,直接通過上面的方法C,B類中的方法C是接受一個(gè)接口類型的參數(shù),那么只需要在C方法中,用這個(gè)接口類型的參數(shù)去調(diào)用D方法,就實(shí)現(xiàn)了在B類中反過來調(diào)用A類中的方法D,這一步稱之為調(diào)用回調(diào)接口。
這也就實(shí)現(xiàn)了B類的C方法中,需要反過來再調(diào)用A類中的D方法,這就是回調(diào)。A調(diào)用B是直調(diào),可以看成高層的代碼用底層的API,我們經(jīng)常這樣寫程序。B調(diào)用A就是回調(diào),底層API需要高層的代碼來執(zhí)行。
最后,總結(jié)一下,回調(diào)方法的步驟:
- A類實(shí)現(xiàn)接口CallBack callback
- A類中包含了一個(gè)B的引用
- B中有一個(gè)參數(shù)為CallBack的方法f(CallBack callback)
- 在A類中調(diào)用B的方法f(CallBack callback)——注冊(cè)回調(diào)接口
- B就可以在f(CallBack callback)方法中調(diào)用A的方法——調(diào)用回調(diào)接口
回調(diào)的例子
我們以一個(gè)兒子在玩游戲,等媽媽把飯做好在通知兒子來吃為例,按照上面的步驟去寫回調(diào);
上面的例子中,顯然應(yīng)該兒子來實(shí)現(xiàn)回調(diào)接口,母親調(diào)用回調(diào)接口。所以我們先定義一個(gè)回調(diào)接口,然后讓兒子去實(shí)現(xiàn)這個(gè)回調(diào)接口。
其代碼如下:
public interface CallBack { void eat(); }
public class Son implements CallBack{ private Mom mom; //A類持有對(duì)B類的引用 public void setMom(Mom mom){ this.mom = mom; } @Override public void eat() { System.out.println("我來吃飯了"); } public void askMom(){ //通過B類的引用調(diào)用含有接口參數(shù)的方法。 System.out.println("飯做了嗎?"); System.out.println("沒做好,我玩游戲了"); new Thread(() -> mom.doCook(Son.this)).start(); System.out.println("玩游戲了中......"); } }
然后我們還需要定義一個(gè)母親的類,里面有一個(gè)含有接口參數(shù)的方法doCook
public class Mom { //在含有接口參數(shù)的方法中利用接口參數(shù)調(diào)用回調(diào)方法 public void doCook(CallBack callBack){ new Thread(new Runnable() { @Override public void run() { try { System.out.println("做飯中......"); Thread.sleep(5000); callBack.eat(); } catch (InterruptedException e) { e.printStackTrace(); } } }).start(); } }
我們通過一個(gè)測(cè)試類:
public class Test { public static void main(String[] args) { Mom mom = new Mom(); Son son = new Son(); son.setMom(mom); son.askMom(); } }
這個(gè)例子就是典型的回調(diào)的例子。Son類實(shí)現(xiàn)了接口的回調(diào)方法,通過askMom這個(gè)方法調(diào)用Mom類中的doCook,實(shí)現(xiàn)注冊(cè)回調(diào)接口,相當(dāng)于A類中調(diào)用B類的代碼C。在Mom類中的doCook來回調(diào)Son類中的eat,來告訴Son類中的結(jié)果。
這樣,我們就實(shí)現(xiàn)了一個(gè)簡(jiǎn)單的,符合定義的回調(diào)。
回調(diào)例子的進(jìn)一步探索
我們主要看一下Son類的代碼:
public class Son implements CallBack{ public Mom mom; public Son(Mom mom){ this.mom = mom; } public void askMom(){ System.out.println("飯做了嗎?"); System.out.println("沒做好,我玩游戲了"); new Thread(() -> mom.doCook(Son.this)).start(); System.out.println("玩游戲了中......"); } @Override public void eat() { System.out.println("好了,我來吃飯了"); } }
這個(gè)類里面,除了輸出一些語(yǔ)句之外,真正有用的部分是mom.doCook(Son.this)以及重寫eat方法。所以,我們可以通過匿名內(nèi)部類的形式,簡(jiǎn)寫這個(gè)回調(diào)。其代碼如下:
public class CallBackTest { public static void main(String[] args) { Mom mom = new Mom(); new Thread(()-> mom.doCook(() -> System.out.println("吃飯了......"))).start(); } }
取消Son類,直接在主方法中通過匿名內(nèi)部類去實(shí)現(xiàn)eat方法。其實(shí)匿名內(nèi)部類就是回調(diào)的體現(xiàn)。
異步回調(diào)與同步回調(diào)
回調(diào)上面我們講了 就是A調(diào)用B類中的方法C,然后在方法C里面通過A類的對(duì)象去調(diào)用A類中的方法D。
我們?cè)谡f一下異步與同步,先說同步的概念
同步
同步指的是在調(diào)用方法的時(shí)候,如果上一個(gè)方法調(diào)用沒有執(zhí)行完,是無法進(jìn)行新的方法調(diào)用。也就是說事情必須一件事情一件事情的做,做完上一件,才能做下一件。
異步
異步相對(duì)于同步,可以不需要等上個(gè)方法調(diào)用結(jié)束,才調(diào)用新的方法。所以,在異步的方法調(diào)用中,是需要一個(gè)方法來通知使用者方法調(diào)用結(jié)果的。
實(shí)現(xiàn)異步的方式
在Java中最常實(shí)現(xiàn)的異步方式就是讓你想異步的方法在一個(gè)新線程中執(zhí)行。
我們會(huì)發(fā)現(xiàn)一點(diǎn),異步方法調(diào)用中需要一個(gè)方法來通知使用者調(diào)用結(jié)果,結(jié)合上面所講,我們會(huì)發(fā)現(xiàn)回調(diào)方法就適合做這個(gè)事情,通過回調(diào)方法來通知使用者調(diào)用的結(jié)果。
那異步回調(diào)就是A調(diào)用B的方法C時(shí)是在一個(gè)新線程當(dāng)中去做的。
上面的母親通知兒子吃飯的例子,就是一個(gè)異步回調(diào)的例子。在一個(gè)新線程中,調(diào)用doCook方法,最后通過eat來接受返回值,當(dāng)然使用lamdba優(yōu)化之后的,本質(zhì)是一樣的。
同步回調(diào)就是A調(diào)用B的方法C沒有在一個(gè)新線程,在執(zhí)行這個(gè)方法C的時(shí)候,我們什么都不能做,只能等待他執(zhí)行完成。
同步回調(diào)與異步回調(diào)的例子
我們看一個(gè)Android中的一個(gè)同步回調(diào)的例子:
button.setOnClickListener(new View.OnClickListener() { @Override public void onClick(View v) { Log.i("button","被點(diǎn)擊"); } });
button通過setOnClickListener注冊(cè)回調(diào)函數(shù),和上面寫的一樣,通過匿名內(nèi)部類的形式將接口的引用傳進(jìn)去。由于button調(diào)用setOnClickListener沒有新建一個(gè)線程,所以這個(gè)是同步的回調(diào)。
而異步回調(diào),就是我們開篇講的那個(gè)例子:
call.enqueue(new Callback<HistoryBean>() { @Override public void onResponse(Call<HistoryBean> call, Response<HistoryBean> response) { HistoryBean hb = response.body(); if(hb == null) return; showText.append(hb.isError() + ""); for(HistoryBean.ResultsBean rb : hb.getResults()){ showText.append(rb.getTitle() + "/n"); } } @Override public void onFailure(Call<HistoryBean> call, Throwable t) { } });
這個(gè)enqueue方法是一個(gè)異步方法去請(qǐng)求遠(yuǎn)程的網(wǎng)絡(luò)數(shù)據(jù)。其內(nèi)部實(shí)現(xiàn)的時(shí)候是通過一個(gè)新線程去執(zhí)行的。
通過這兩個(gè)例子,我們可以看出同步回調(diào)與異步回調(diào)的使用其實(shí)是根據(jù)不同的需求而設(shè)計(jì)。不能說一種取代另一種,像上面的按鈕點(diǎn)擊事件中,如果是異步回調(diào),用戶點(diǎn)擊按鈕之后其點(diǎn)擊效果不是馬上出現(xiàn),而用戶又不會(huì)執(zhí)行其他操作,那么會(huì)感覺很奇怪。而像網(wǎng)絡(luò)請(qǐng)求的異步回調(diào),因?yàn)槭芟抻谡?qǐng)求資源可能不存在,網(wǎng)絡(luò)連接不穩(wěn)定等等原因?qū)е掠脩舨磺宄椒▓?zhí)行的時(shí)候,所以會(huì)用異步回調(diào),發(fā)起方法調(diào)用之后去做其他事情,然后等回調(diào)的通知。
回調(diào)方法在通信中的應(yīng)用
上面提到的回調(diào)方法,除了網(wǎng)絡(luò)請(qǐng)求框架的回調(diào)除外,其回調(diào)方法都是沒有參數(shù),下面,我們看一下在回調(diào)方法中加入?yún)?shù)來實(shí)現(xiàn)一些通信問題。
如果我們想要A類得到B類經(jīng)過一系列計(jì)算,處理后數(shù)據(jù),而且兩個(gè)類是不能通過簡(jiǎn)單的將B的引用給A類就可以得到數(shù)據(jù)的。我們可以考慮回調(diào)。
步驟如下:
- 在擁有數(shù)據(jù)的那個(gè)類里面寫一個(gè)回調(diào)的接口。-->這里就是B類中寫一個(gè)回調(diào)接口
- 回調(diào)方法接收一個(gè)參數(shù),這個(gè)參數(shù)就是要得到的數(shù)據(jù)
- 同樣是在這個(gè)類里寫一個(gè)注冊(cè)回調(diào)的方法。
- 在注冊(cè)回調(diào)方法,用接口的引用去調(diào)用回調(diào)接口,把B類的數(shù)據(jù)當(dāng)做參數(shù)傳入回調(diào)的方法中。
- 在A類中,用B類的引用去注冊(cè)回調(diào)接口,把B類中的數(shù)據(jù)通過回調(diào)傳到A類中。
上面說的步驟,有點(diǎn)抽象。下面我們看一個(gè)例子,一個(gè)是Client,一個(gè)是Server。Client去請(qǐng)求Server經(jīng)過耗時(shí)處理后的數(shù)據(jù)。
public class Client{ public Server server; public String request; //鏈接Server,得到Server引用。 public Client connect(Server server){ this.server = server; return this; } //Client,設(shè)置request public Client setRequest(String request){ this.request = request; return this; } //異步發(fā)送請(qǐng)求的方法,lamdba表達(dá)式。 public void enqueue(Server.CallBack callBack){ new Thread(()->server.setCallBack(request,callBack)).start(); } }
public class Server { public String response = "這是一個(gè)html"; //注冊(cè)回調(diào)接口的方法,把數(shù)據(jù)通過參數(shù)傳給回調(diào)接口 public void setCallBack(String request,CallBack callBack){ System.out.println("已經(jīng)收到request,正在計(jì)算當(dāng)中......"); new Thread(() -> { try { Thread.sleep(5000); callBack.onResponse(request + response); } catch (InterruptedException e) { e.printStackTrace(); callBack.onFail(e); } }).start(); } //在擁有數(shù)據(jù)的那個(gè)類里面寫一個(gè)接口 public interface CallBack{ void onResponse(String response); void onFail(Throwable throwable); } }
接下來,我們看一下測(cè)試的例子:
public class CallBackTest { public static void main(String[] args) { Client client = new Client(); client.connect(new Server()).setRequest("這個(gè)文件是什么?").enqueue(new Server.CallBack() { @Override public void onResponse(String response) { System.out.println(response); } @Override public void onFail(Throwable throwable) { System.out.println(throwable.getMessage()); } }); } }
結(jié)果如下:
已經(jīng)收到request,正在計(jì)算當(dāng)中...... 這個(gè)文件是什么?這是一個(gè)html
以上就是通過回調(diào)的方式進(jìn)行通信。
以上就是本文的全部?jī)?nèi)容,希望本文的內(nèi)容對(duì)大家的學(xué)習(xí)或者工作能帶來一定的幫助,同時(shí)也希望多多支持腳本之家!
相關(guān)文章
java實(shí)現(xiàn)字符串匹配求兩個(gè)字符串的最大公共子串
這篇文章主要介紹了java實(shí)現(xiàn)求兩個(gè)字符串最大公共子串的方法,詳細(xì)的描述了兩個(gè)字符串的最大公共子串算法的實(shí)現(xiàn),需要的朋友可以參考下2016-10-10Spring?Data?JPA系列QueryByExampleExecutor使用詳解
這篇文章主要為大家介紹了Spring?Data?JPA系列QueryByExampleExecutor使用示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-09-09淺談java中BigDecimal的equals與compareTo的區(qū)別
下面小編就為大家?guī)硪黄獪\談java中BigDecimal的equals與compareTo的區(qū)別。小編覺得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧2016-11-11SpringBoot項(xiàng)目微信云托管入門部署實(shí)踐
本文主要介紹了SpringBoot項(xiàng)目微信云托管入門部署實(shí)踐,文中通過示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-03-03Mybatis插入時(shí)返回自增主鍵方式(selectKey和useGeneratedKeys)
這篇文章主要介紹了Mybatis插入時(shí)返回自增主鍵方式(selectKey和useGeneratedKeys),具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-09-09Java將字節(jié)轉(zhuǎn)換為十六進(jìn)制代碼分享
我們知道,在java中,一個(gè)byte 就是一個(gè)字節(jié),也就是八個(gè)二進(jìn)制位;而4個(gè)二進(jìn)制位就可以表示一個(gè)十六進(jìn)制位,所以一個(gè)byte可以轉(zhuǎn)化為2個(gè)十六進(jìn)制位。下面我們就來詳細(xì)看下具體方法吧。2016-01-01