成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

兩個經(jīng)典例子讓你徹底理解Java回調(diào)機制

開發(fā) 后端
先讓我們通過一個生活中的場景來還原一下回調(diào)的場景:你遇到了一個技術難題(比如,1+1等于幾?太難了!),于是你去咨詢大牛,大牛說現(xiàn)在正在忙,待會兒告訴你結果。

本文轉載自微信公眾號「程序新視界」,作者二師兄。轉載本文請聯(lián)系程序新視界公眾號。   

 前言

先讓我們通過一個生活中的場景來還原一下回調(diào)的場景:你遇到了一個技術難題(比如,1+1等于幾?太難了!),于是你去咨詢大牛,大牛說現(xiàn)在正在忙,待會兒告訴你結果。

此時,你可能會去刷朋友圈了,等大牛忙完之后,告訴你答案是2。

那么,這個過程中詢問問題(調(diào)用對方接口),然后問題解決之后再告訴你(對方處理完再調(diào)用你,通知結果),這一過程便是回調(diào)。

系統(tǒng)調(diào)用的分類

應用系統(tǒng)模塊之間的調(diào)用,通常分為:同步調(diào)用,異步調(diào)用,回調(diào)。

同步調(diào)用

同步調(diào)用是最基本的調(diào)用方式。類A的a()方法調(diào)用類B的b()方法,類A的方法需要等到B類的方法執(zhí)行完成才會繼續(xù)執(zhí)行。如果B的方法長時間阻塞,就會導致A類方法無法正常執(zhí)行下去。

異步調(diào)用

如果A調(diào)用B,B的執(zhí)行時間比較長,那么就需要考慮進行異步處理,使得B的執(zhí)行不影響A。通常在A中新起一個線程用來調(diào)用B,然后A中的代碼繼續(xù)執(zhí)行。

異步通常分兩種情況:第一,不需要調(diào)用結果,直接調(diào)用即可,比如發(fā)送消息通知;第二,需要異步調(diào)用結果,在Java中可使用Future+Callable實現(xiàn)。

回調(diào)

通過上圖我們可以看到回到屬于一種雙向的調(diào)用方式。回調(diào)的基本上思路是:A調(diào)用B,B處理完之后再調(diào)用A提供的回調(diào)方法(通常為callbakc())通知結果。

通常回調(diào)分為:同步回調(diào)和異步回調(diào)。網(wǎng)絡上大多數(shù)的回調(diào)案例都是同步回調(diào)。

其中同步回調(diào)與同步調(diào)用類似,代碼運行到某一個位置的時候,如果遇到了需要回調(diào)的代碼,會在這里等待,等待回調(diào)結果返回后再繼續(xù)執(zhí)行。

而異步回調(diào)與異步調(diào)用類似,代碼執(zhí)行到需要回調(diào)的代碼的時候,并不會停下來,而是繼續(xù)執(zhí)行,當然可能過一會回調(diào)的結果會返回回來。

同步回調(diào)實例

下面我們以同步回調(diào)為例來講解回調(diào)的Java代碼實現(xiàn)。整個過程就模擬上面問答問題的場景。

首先,定義給一個CallBack的接口,將回調(diào)的功能進行單獨抽離:

  1. public interface CallBack { 
  2.     void callback(String string); 

CallBack接口中提供了一個callback方法,用于回調(diào)時調(diào)用。

然后定義問問題的人Person:

  1. public class Person implements CallBack { 
  2.  
  3.     private Genius genius; 
  4.  
  5.     public Person(Genius genius) { 
  6.         this.genius = genius; 
  7.     } 
  8.  
  9.     @Override 
  10.     public void callback(String string) { 
  11.         System.out.println("收到答案:" + string); 
  12.     } 
  13.  
  14.     public void ask() { 
  15.         genius.answer(this); 
  16.     } 
  17.  

由于Person要提供回調(diào)方法,因此實現(xiàn)CallBack接口及其方法,方法中主要針對回調(diào)結果進行處理。

同時,由于Person要調(diào)用Genius對應的方法,因此要持有Genius的引用,這里通過構造方法傳入。

定義回答問題的大神Genius類:

  1. public class Genius { 
  2.  
  3.     public void answer(CallBack callBack) { 
  4.         System.out.println("在忙其他事..."); 
  5.         try { 
  6.             Thread.sleep(2000); 
  7.             System.out.println("忙完其他事,開始計算..."); 
  8.         } catch (InterruptedException e) { 
  9.             e.printStackTrace(); 
  10.         } 
  11.  
  12.         System.out.println("天才計算出答案為:2"); 
  13.         // 回調(diào)告訴你 
  14.         callBack.callback("2"); 
  15.     } 

這模擬大神正在忙碌,線程睡眠2秒,忙碌完之后,開始幫忙計算答案,獲得答案之后,調(diào)用CallBack接口的callback方法進行回調(diào),通知結果。

通過Main方法進行測試:

  1. public static void main(String[] args) { 
  2.     Genius genius = new Genius(); 
  3.     Person you = new Person(genius); 
  4.     you.ask(); 

執(zhí)行打印結果如下:

  1. 在忙其他事... 
  2. 忙完其他事,開始計算... 
  3. 天才計算出答案為:2 
  4. 收到答案:2 

上面的過程,就實現(xiàn)了一個同步回調(diào)的功能。當然,從程序設計上來說,可以對Person和Genius進一步抽象化處理,通過接口的形式呈現(xiàn)。

在上述回調(diào)機制的代碼實現(xiàn)中,最核心的是在調(diào)用answer方法時傳遞了this參數(shù),即調(diào)用者自身。

從本質(zhì)上來說,回調(diào)是一種思想,是一種機制,至于具體如何實現(xiàn),如何通過代碼將回調(diào)實現(xiàn)得優(yōu)雅、實現(xiàn)得可擴展性比較高,就需要八仙過海各顯神通了。

異步回調(diào)實例

上面的實例演示了同步回調(diào),很明顯在調(diào)用的過受到Genius執(zhí)行時長的影響,需要等到Genius處理完才能繼續(xù)執(zhí)行Person方法中的后續(xù)代碼。

下面在上述示例上進行改進,Person提供一個支持異步回調(diào)的方法:

  1.  public void askASyn() { 
  2.     System.out.println("創(chuàng)建新線程請教問題"); 
  3.     new Thread(() -> genius.answer(this)).start(); 
  4.     System.out.println("新線程已啟動..."); 

在該方法內(nèi),新建了一個線程用來處理Genius#answer方法的調(diào)用,這樣就能夠跳過Genius#answer方法的阻塞,直接執(zhí)行下面的操作(日志打印)。

在main方法中將調(diào)用的方法改為askASyn,打印結果如下:

  1. 創(chuàng)建新線程請教問題 
  2. 新線程已啟動... 
  3. 在忙其他事... 
  4. 忙完其他事,開始計算... 
  5. 天才計算出答案為:2 
  6. 收到答案:2 

可以看出,直接打印了“新線程已啟動...”,后續(xù)才打印出Genius#answer方法方法中處理日志和回調(diào)時callback方法接收到的信息。

基于Future的半異步

除了上述的同步,異步處理,還有一種介于同步和異步之間的基于Future的半異步處理。

在Java使用nio后無法立即拿到真實的數(shù)據(jù),而是先得到一個"future",可以理解為郵戳或快遞單,為了獲悉真正的數(shù)據(jù)我們需要不停的通過快遞單號"future"查詢快遞是否真正寄到。

Futures是一個抽象的概念,它表示一個值,在某一點會變得可用。一個Future要么獲得計算完的結果,要么獲得計算失敗后的異常。

通常什么時候會用到Future呢?一般來說,當執(zhí)行一個耗時的任務時,使用Future就可以讓線程暫時去處理其他的任務,等長任務執(zhí)行完畢再返回其結果。

經(jīng)常會使用到Future的場景有:1. 計算密集場景。2. 處理大數(shù)據(jù)量。3. 遠程方法調(diào)用等。

Java在java.util.concurrent包中附帶了Future接口,它使用Executor異步執(zhí)行。

例如下面的代碼,每傳遞一個Runnable對象到ExecutorService.submit()方法就會得到一個回調(diào)的Future,使用它檢測是否執(zhí)行,這種方法可以是同步等待線處理結果完成。

  1. public class TestFuture { 
  2.  
  3.     public static void main(String[] args) { 
  4.  
  5.         //實現(xiàn)一個Callable接口 
  6.         Callable<User> c = () -> { 
  7.             //這里是業(yè)務邏輯處理 
  8.  
  9.             //讓當前線程阻塞1秒看下效果 
  10.             Thread.sleep(1000); 
  11.             return new User("張三"); 
  12.         }; 
  13.  
  14.         ExecutorService es = Executors.newFixedThreadPool(2); 
  15.  
  16.         // 記得要用submit,執(zhí)行Callable對象 
  17.         Future<User> fn = es.submit(c); 
  18.         // 一定要調(diào)用這個方法,不然executorService.isTerminated()永遠不為true 
  19.         es.shutdown(); 
  20.         // 無限循環(huán)等待任務處理完畢  如果已經(jīng)處理完畢 isDone返回true 
  21.         while (!fn.isDone()) { 
  22.             try { 
  23.                 //處理完畢后返回的結果 
  24.                 User nt = fn.get(); 
  25.                 System.out.println(nt.name); 
  26.             } catch (InterruptedException | ExecutionException e) { 
  27.                 e.printStackTrace(); 
  28.             } 
  29.         } 
  30.     } 
  31.  
  32.     static class User { 
  33.         private String name
  34.  
  35.         private User(String name) { 
  36.             this.name = name
  37.         } 
  38.     } 

此種情況下雖然是創(chuàng)建了新線程來進行處理,但還是需要等待處理的結果。好處是可以將批量的處理,分為幾個線程同時進行處理,最后對結果進行合并,達到提升處理效率的目的。

小結

經(jīng)過這篇文章,想必大家對Java的回調(diào)機制已經(jīng)有所了解,在各類開源框架中,其實也會經(jīng)常看到回調(diào)的使用,活學活用。

 

責任編輯:武曉燕 來源: 程序新視界
相關推薦

2020-11-03 10:32:48

回調(diào)函數(shù)模塊

2021-04-07 13:28:21

函數(shù)程序員異步

2020-12-17 10:38:33

回調(diào)函數(shù)make_youtia函數(shù)

2009-11-06 16:05:37

WCF回調(diào)契約

2010-09-10 15:26:05

SOAP封裝

2012-02-01 10:33:59

Java

2020-04-27 09:40:13

Reacthooks前端

2015-10-26 09:25:42

2023-10-11 08:18:22

RocketMQ邏輯訂閱

2024-01-19 13:45:00

Pandas代碼深度學習

2024-03-15 08:23:26

異步編程函數(shù)

2025-05-07 01:20:11

SpringMVC核心機制

2024-06-21 08:32:24

2020-12-08 08:14:11

SQL注入數(shù)據(jù)庫

2019-11-05 10:03:08

callback回調(diào)函數(shù)javascript

2020-10-14 09:11:44

IO 多路復用實現(xiàn)機

2022-07-13 15:46:57

Python數(shù)據(jù)可視化代碼片段

2022-12-13 18:09:25

連接狀態(tài)客戶端

2024-01-17 08:18:14

RPAJava技術

2024-02-02 09:00:14

內(nèi)存泄漏對象
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 国内精品免费久久久久软件老师 | 日韩欧美亚洲一区 | 日韩网| 日韩欧美在线免费观看 | 欧美在线视频网 | 亚洲一区国产精品 | 中文字幕一区二区三区四区五区 | 人人干超碰 | 91久久久久 | 国产电影一区二区 | 日本一区二区高清不卡 | av永久免费 | 污视频在线免费观看 | 黄色小视频大全 | 亚洲欧美久久 | xxxxxx国产| 国产精品久久久免费 | 特级生活片 | 欧美精品一区二区三区四区 在线 | 五月婷婷 六月丁香 | 日本精品一区二区在线观看 | 午夜激情在线视频 | 日韩中文字幕网 | 久久久久久国产精品久久 | 精品国产乱码久久久久久老虎 | 久久99精品久久久久久国产越南 | 国产成人a亚洲精品 | 国产美女视频黄a视频免费 国产精品福利视频 | 中文字幕久久精品 | 国产精品高潮呻吟久久av野狼 | 欧美色欧美亚洲另类七区 | 国产美女自拍视频 | 日韩欧美在线观看 | 久久精品欧美电影 | 亚洲一区二区中文字幕 | 国产四区 | 久久久噜噜噜www成人网 | 日韩综合网 | 黄色毛片网站在线观看 | 国产色婷婷精品综合在线手机播放 | 一级全黄少妇性色生活免费看 |