詳解java 回調(diào)機制
代碼示例
接下來看一下回調(diào)的代碼示例,代碼模擬的是這樣一種場景:老師問學(xué)生問題,學(xué)生思考完畢回答老師。
首先定義一個回調(diào)接口,只有一個方法tellAnswer(int answer),即學(xué)生思考完畢告訴老師答案:
/** * 回調(diào)接口 */publicinterfaceCallback{publicvoidtellAnswer( intanswer); }
定義一個老師對象,實現(xiàn)Callback接口:
/** * 老師對象 */publicclassTeacherimplementsCallback{privateStudent student;publicTeacher(Student student) { this.student = student; } publicvoidaskQuestion() { student.resolveQuestion( this); } @OverridepublicvoidtellAnswer( intanswer) { System.out.println( “知道了,你的答案是”+ answer); } }
老師對象有兩個public方法:
(1)回調(diào)接口tellAnswer(int answer),即學(xué)生回答完畢問題之后,老師要做的事情
(2)問問題方法askQuestion(),即向?qū)W生問問題
接著定義一個學(xué)生接口,學(xué)生當(dāng)然是解決問題,但是接收一個Callback參數(shù),這樣學(xué)生就知道解決完畢問題向誰報告:
/** * 學(xué)生接口,Java學(xué)習(xí)交流QQ群:589809992 我們一起學(xué)Java! */publicinterfaceStudent{publicvoidresolveQuestion(Callback callback); }
最后定義一個具體的學(xué)生叫Ricky:
/** * 一個名叫Ricky的同學(xué)解決老師提出的問題 */publicclassRickyimplementsStudent{@OverridepublicvoidresolveQuestion(Callback callback) { // 模擬解決問題try{ Thread.sleep( 3000); } catch(InterruptedException e) { } // 回調(diào),告訴老師作業(yè)寫了多久callback.tellAnswer( 3); } }
在解決完畢問題之后,第16行向老師報告答案。
寫一個測試類,比較簡單:
/** * 回調(diào)測試,Java學(xué)習(xí)交流QQ群:589809992 我們一起學(xué)Java! */publicclassCallbackTest{@TestpublicvoidtestCallback() { Student student = newRicky(); Teacher teacher = newTeacher(student); teacher.askQuestion(); } }
代碼運行結(jié)果就一行:
知道了,你的答案是3
簡單總結(jié)、分析一下這個例子就是:
(1)老師調(diào)用學(xué)生接口的方法resolveQuestion,向?qū)W生提問
(2)學(xué)生解決完畢問題之后調(diào)用老師的回調(diào)方法tellAnswer
這樣一套流程,構(gòu)成了一種雙向調(diào)用的關(guān)系。
代碼分析
分析一下上面的代碼,上面的代碼我這里做了兩層的抽象:
(1)將老師進(jìn)行抽象
將老師進(jìn)行抽象之后,對于學(xué)生來說,就不需要關(guān)心到底是哪位老師詢問我問題,只要我根據(jù)詢問的問題,得出答案,然后告訴提問的老師就可以了,即使老師換了一茬又一茬,對我學(xué)生而言都是沒有任何影響的
(2)將學(xué)生進(jìn)行抽象
將學(xué)生進(jìn)行抽象之后,對于老師這邊來說就非常靈活,因為老師未必對一個學(xué)生進(jìn)行提問,可能同時對Ricky、Jack、Lucy三個學(xué)生進(jìn)行提問,這樣就可以將成員變量Student改為List,這樣在提問的時候遍歷Student列表進(jìn)行提問,然后得到每個學(xué)生的回答即可
這個例子是一個典型的體現(xiàn)接口作用的例子,之所以這么說是因為我想到有些朋友可能不太明白接口的好處,不太明白接口好處的朋友可以重點看一下這個例子,多多理解。
總結(jié)起來,回調(diào)的核心就是回調(diào)方將本身即this傳遞給調(diào)用方,這樣調(diào)用方就可以在調(diào)用完畢之后告訴回調(diào)方它想要知道的信息。回調(diào)是一種思想、是一種機制,至于具體如何實現(xiàn),如何通過代碼將回調(diào)實現(xiàn)得優(yōu)雅、實現(xiàn)得可擴(kuò)展性比較高,一看開發(fā)者的個人水平,二看開發(fā)者對業(yè)務(wù)的理解程度。
同步回調(diào)與異步回調(diào)
上面的例子,可能有人會提出這樣的疑問:
這個例子需要用什么回調(diào)啊,使用同步調(diào)用的方式,學(xué)生對象回答完畢問題之后直接把回答的答案返回給老師對象不就好了?
這個問題的提出沒有任何問題,可以從兩個角度去理解這個問題。
首先,老師不僅僅想要得到學(xué)生的答案怎么辦?可能這個老師是個更喜歡聽學(xué)生解題思路的老師,在得到學(xué)生的答案之前,老師更想先知道學(xué)生姓名和學(xué)生的解題思路,當(dāng)然有些人可以說,那我可以定義一個對象,里面加上學(xué)生的姓名和解題思路不就好了。這個說法在我看來有兩個問題:
(1)如果老師想要的數(shù)據(jù)越來越多,那么返回的對象得越來越大,而使用回調(diào)則可以進(jìn)行數(shù)據(jù)分離,將一批數(shù)據(jù)放在回調(diào)方法中進(jìn)行處理,至于哪些數(shù)據(jù)依具體業(yè)務(wù)而定,如果需要增加返回參數(shù),直接在回調(diào)方法中增加即可
(2)無法解決老師希望得到學(xué)生姓名、學(xué)生解題思路先于學(xué)生回答的答案的問題
因此我認(rèn)為簡單的返回某個結(jié)果確實沒有必要使用回調(diào)而可以直接使用同步調(diào)用,但是如果有多種數(shù)據(jù)需要處理且數(shù)據(jù)有主次之分,使用回調(diào)會是一種更加合適的選擇,優(yōu)先處理的數(shù)據(jù)放在回調(diào)方法中先處理掉。
另外一個理解的角度則更加重要,就是標(biāo)題說的同步回調(diào)和異步回調(diào)了。例子是一個同步回調(diào)的例子,意思是老師向Ricky問問題,Ricky給出答案,老師問下一個同學(xué),得到答案之后繼續(xù)問下一個同學(xué),這是一種正常的場景,但是如果我把場景改一下:
老師并不想One-By-One這樣提問,而是同時向Ricky、Mike、Lucy、Bruce、Kate五位同學(xué)提問,讓同學(xué)們自己思考,哪位同學(xué)思考好了就直接告訴老師答案即可。
這種場景相當(dāng)于是說,同學(xué)思考完畢完畢問題要有一個辦法告訴老師,有兩個解決方案:
(1)使用Future+Callable的方式,等待異步線程執(zhí)行結(jié)果,這相當(dāng)于就是同步調(diào)用的一種變種,因為其本質(zhì)還是方法返回一個結(jié)果,即學(xué)生的回答
(2)使用異步回調(diào),同學(xué)回答完畢問題,調(diào)用回調(diào)接口方法告訴老師答案即可。由于老師對象被抽象成了Callback接口,因此這種做法的擴(kuò)展性非常好,就像之前說的,即使老師換了換了一茬又一茬,對于同學(xué)來說,只關(guān)心的是調(diào)用Callback接口回傳必要的信息即可
非常好我支持^.^
(0) 0%
不好我反對
(0) 0%