溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務(wù)條款》

深入淺析Java中的回調(diào)機(jī)制

發(fā)布時間:2020-11-10 17:05:25 來源:億速云 閱讀:142 作者:Leah 欄目:編程語言

本篇文章給大家分享的是有關(guān)深入淺析Java中的回調(diào)機(jī)制,小編覺得挺實用的,因此分享給大家學(xué)習(xí),希望大家閱讀完這篇文章后可以有所收獲,話不多說,跟著小編一起來看看吧。

模塊間調(diào)用

在一個應(yīng)用系統(tǒng)中,無論使用何種語言開發(fā),必然存在模塊之間的調(diào)用,調(diào)用的方式分為幾種:

(1)同步調(diào)用

深入淺析Java中的回調(diào)機(jī)制

同步調(diào)用是最基本并且最簡單的一種調(diào)用方式,類A的方法a()調(diào)用類B的方法b(),一直等待b()方法執(zhí)行完畢,a()方法繼續(xù)往下走。這種調(diào)用方式適用于方法b()執(zhí)行時間不長的情況,因為b()方法執(zhí)行時間一長或者直接阻塞的話,a()方法的余下代碼是無法執(zhí)行下去的,這樣會造成整個流程的阻塞。

(2)異步調(diào)用

深入淺析Java中的回調(diào)機(jī)制

異步調(diào)用是為了解決同步調(diào)用可能出現(xiàn)阻塞,導(dǎo)致整個流程卡住而產(chǎn)生的一種調(diào)用方式。類A的方法方法a()通過新起線程的方式調(diào)用類B的方法b(),代碼接著直接往下執(zhí)行,這樣無論方法b()執(zhí)行時間多久,都不會阻塞住方法a()的執(zhí)行。但是這種方式,由于方法a()不等待方法b()的執(zhí)行完成,在方法a()需要方法b()執(zhí)行結(jié)果的情況下(視具體業(yè)務(wù)而定,有些業(yè)務(wù)比如啟異步線程發(fā)個微信通知、刷新一個緩存這種就沒必要),必須通過一定的方式對方法b()的執(zhí)行結(jié)果進(jìn)行監(jiān)聽。在Java中,可以使用Future+Callable的方式做到這一點。

(3)回調(diào)

深入淺析Java中的回調(diào)機(jī)制

最后是回調(diào),回調(diào)的思想是:

  • 類A的a()方法調(diào)用類B的b()方法
  • 類B的b()方法執(zhí)行完畢主動調(diào)用類A的callback()方法

這樣一種調(diào)用方式組成了上圖,也就是一種雙向的調(diào)用方式。

代碼示例

接下來看一下回調(diào)的代碼示例,代碼模擬的是這樣一種場景:老師問學(xué)生問題,學(xué)生思考完畢回答老師。

首先定義一個回調(diào)接口,只有一個方法tellAnswer(int answer),即學(xué)生思考完畢告訴老師答案:

/**
 * 回調(diào)接口,原文出處http://www.cnblogs.com/xrq730/p/6424471.html
 */
public interface Callback {
 public void tellAnswer(int answer);
}

定義一個老師對象,實現(xiàn)Callback接口:

/**
 * 老師對象,原文出處http://www.cnblogs.com/xrq730/p/6424471.html
 */
public class Teacher implements Callback {
 private Student student;
 public Teacher(Student student) {
 this.student = student;
 }
 public void askQuestion() {
 student.resolveQuestion(this);
 }
 @Override
 public void tellAnswer(int answer) {
 System.out.println("知道了,你的答案是" + answer);
 }
}

老師對象有兩個public方法:

(1)回調(diào)接口tellAnswer(int answer),即學(xué)生回答完畢問題之后,老師要做的事情

(2)問問題方法askQuestion(),即向?qū)W生問問題

接著定義一個學(xué)生接口,學(xué)生當(dāng)然是解決問題,但是接收一個Callback參數(shù),這樣學(xué)生就知道解決完畢問題向誰報告:

/**
 * 學(xué)生接口,原文出處http://www.cnblogs.com/xrq730/p/6424471.html
 */
public interface Student {
 public void resolveQuestion(Callback callback); 
}

最后定義一個具體的學(xué)生叫Ricky:

/**
 * 一個名叫Ricky的同學(xué)解決老師提出的問題,原文出處http://www.cnblogs.com/xrq730/p/6424471.html
 */
public class Ricky implements Student {
 @Override
 public void resolveQuestion(Callback callback) {
 // 模擬解決問題
 try {
  Thread.sleep(3000);
 } catch (InterruptedException e) {  
 }
 // 回調(diào),告訴老師作業(yè)寫了多久
 callback.tellAnswer(3);
 }
}

在解決完畢問題之后,第16行向老師報告答案。

寫一個測試類,比較簡單:

/**
 * 回調(diào)測試,原文出處http://www.cnblogs.com/xrq730/p/6424471.html
 */
public class CallbackTest {

 @Test
 public void testCallback() {
 Student student = new Ricky();
 Teacher teacher = new Teacher(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>,這樣在提問的時候遍歷Student列表進(jìn)行提問,然后得到每個學(xué)生的回答即可

這個例子是一個典型的體現(xiàn)接口作用的例子,之所以這么說是因為我想到有些朋友可能不太明白接口的好處,不太明白接口好處的朋友可以重點看一下這個例子,多多理解。

總結(jié)起來,回調(diào)的核心就是回調(diào)方將本身即this傳遞給調(diào)用方,這樣調(diào)用方就可以在調(diào)用完畢之后告訴回調(diào)方它想要知道的信息?;卣{(diào)是一種思想、是一種機(jī)制,至于具體如何實現(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接口回傳必要的信息即可

以上就是深入淺析Java中的回調(diào)機(jī)制,小編相信有部分知識點可能是我們?nèi)粘9ぷ鲿姷交蛴玫降?。希望你能通過這篇文章學(xué)到更多知識。更多詳情敬請關(guān)注億速云行業(yè)資訊頻道。

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進(jìn)行舉報,并提供相關(guān)證據(jù),一經(jīng)查實,將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI