一区二区三区在线-一区二区三区亚洲视频-一区二区三区亚洲-一区二区三区午夜-一区二区三区四区在线视频-一区二区三区四区在线免费观看

服務(wù)器之家:專(zhuān)注于服務(wù)器技術(shù)及軟件下載分享
分類(lèi)導(dǎo)航

PHP教程|ASP.NET教程|Java教程|ASP教程|編程技術(shù)|正則表達(dá)式|C/C++|IOS|C#|Swift|Android|JavaScript|易語(yǔ)言|

服務(wù)器之家 - 編程語(yǔ)言 - Java教程 - 兩個(gè)例子了解java中的回調(diào)機(jī)制

兩個(gè)例子了解java中的回調(diào)機(jī)制

2021-08-06 10:47程序新視界 Java教程

這篇文章主要介紹了Java中回調(diào)機(jī)制的相關(guān)資料,幫助大家更好的理解和學(xué)習(xí)使用Java,感興趣的朋友可以了解下

前言

 

先讓我們通過(guò)一個(gè)生活中的場(chǎng)景來(lái)還原一下回調(diào)的場(chǎng)景:你遇到了一個(gè)技術(shù)難題(比如,1+1等于幾?太難了!),于是你去咨詢(xún)大牛,大牛說(shuō)現(xiàn)在正在忙,待會(huì)兒告訴你結(jié)果。

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

那么,這個(gè)過(guò)程中詢(xún)問(wèn)問(wèn)題(調(diào)用對(duì)方接口),然后問(wèn)題解決之后再告訴你(對(duì)方處理完再調(diào)用你,通知結(jié)果),這一過(guò)程便是回調(diào)。

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

 

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

兩個(gè)例子了解java中的回調(diào)機(jī)制

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

兩個(gè)例子了解java中的回調(diào)機(jī)制

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

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

兩個(gè)例子了解java中的回調(diào)機(jī)制

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

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

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

而異步回調(diào)與異步調(diào)用類(lèi)似,代碼執(zhí)行到需要回調(diào)的代碼的時(shí)候,并不會(huì)停下來(lái),而是繼續(xù)執(zhí)行,當(dāng)然可能過(guò)一會(huì)回調(diào)的結(jié)果會(huì)返回回來(lái)。

同步回調(diào)實(shí)例

 

下面我們以同步回調(diào)為例來(lái)講解回調(diào)的Java代碼實(shí)現(xiàn)。整個(gè)過(guò)程就模擬上面問(wèn)答問(wèn)題的場(chǎng)景。

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

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

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

然后定義問(wèn)問(wèn)題的人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.  
  18. }

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

同時(shí),由于Person要調(diào)用Genius對(duì)應(yīng)的方法,因此要持有Genius的引用,這里通過(guò)構(gòu)造方法傳入。

定義回答問(wèn)題的大神Genius類(lèi):

  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("忙完其他事,開(kāi)始計(jì)算...");
  8.         } catch (InterruptedException e) {
  9.             e.printStackTrace();
  10.         }
  11.  
  12.         System.out.println("天才計(jì)算出答案為:2");
  13.         // 回調(diào)告訴你
  14.         callBack.callback("2");
  15.     }
  16. }

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

通過(guò)Main方法進(jìn)行測(cè)試:

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

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

  1. 在忙其他事...
  2. 忙完其他事,開(kāi)始計(jì)算...
  3. 天才計(jì)算出答案為:2
  4. 收到答案:2

上面的過(guò)程,就實(shí)現(xiàn)了一個(gè)同步回調(diào)的功能。當(dāng)然,從程序設(shè)計(jì)上來(lái)說(shuō),可以對(duì)Person和Genius進(jìn)一步抽象化處理,通過(guò)接口的形式呈現(xiàn)。

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

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

異步回調(diào)實(shí)例

 

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

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

  1.  public void askASyn() {
  2.     System.out.println("創(chuàng)建新線(xiàn)程請(qǐng)教問(wèn)題");
  3.     new Thread(() -> genius.answer(this)).start();
  4.     System.out.println("新線(xiàn)程已啟動(dòng)...");
  5. }

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

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

  1. 創(chuàng)建新線(xiàn)程請(qǐng)教問(wèn)題
  2. 新線(xiàn)程已啟動(dòng)...
  3. 在忙其他事...
  4. 忙完其他事,開(kāi)始計(jì)算...
  5. 天才計(jì)算出答案為:2
  6. 收到答案:2

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

基于Future的半異步

 

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

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

Futures是一個(gè)抽象的概念,它表示一個(gè)值,在某一點(diǎn)會(huì)變得可用。一個(gè)Future要么獲得計(jì)算完的結(jié)果,要么獲得計(jì)算失敗后的異常。

通常什么時(shí)候會(huì)用到Future呢?一般來(lái)說(shuō),當(dāng)執(zhí)行一個(gè)耗時(shí)的任務(wù)時(shí),使用Future就可以讓線(xiàn)程暫時(shí)去處理其他的任務(wù),等長(zhǎng)任務(wù)執(zhí)行完畢再返回其結(jié)果。

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

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

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

  1. public class TestFuture {
  2.  
  3.     public static void main(String[] args) {
  4.  
  5.         //實(shí)現(xiàn)一個(gè)Callable接口
  6.         Callable<User> c = () -> {
  7.             //這里是業(yè)務(wù)邏輯處理
  8.  
  9.             //讓當(dāng)前線(xiàn)程阻塞1秒看下效果
  10.             Thread.sleep(1000);
  11.             return new User("張三");
  12.         };
  13.  
  14.         ExecutorService es = Executors.newFixedThreadPool(2);
  15.  
  16.         // 記得要用submit,執(zhí)行Callable對(duì)象
  17.         Future<User> fn = es.submit(c);
  18.         // 一定要調(diào)用這個(gè)方法,不然executorService.isTerminated()永遠(yuǎn)不為true
  19.         es.shutdown();
  20.         // 無(wú)限循環(huán)等待任務(wù)處理完畢  如果已經(jīng)處理完畢 isDone返回true
  21.         while (!fn.isDone()) {
  22.             try {
  23.                 //處理完畢后返回的結(jié)果
  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.     }
  39. }

此種情況下雖然是創(chuàng)建了新線(xiàn)程來(lái)進(jìn)行處理,但還是需要等待處理的結(jié)果。好處是可以將批量的處理,分為幾個(gè)線(xiàn)程同時(shí)進(jìn)行處理,最后對(duì)結(jié)果進(jìn)行合并,達(dá)到提升處理效率的目的。

小結(jié)

 

經(jīng)過(guò)這篇文章,想必大家對(duì)Java的回調(diào)機(jī)制已經(jīng)有所了解,在各類(lèi)開(kāi)源框架中,其實(shí)也會(huì)經(jīng)常看到回調(diào)的使用,活學(xué)活用。

以上就是兩個(gè)例子了解java中的回調(diào)機(jī)制的詳細(xì)內(nèi)容,更多關(guān)于Java 回調(diào)機(jī)制的資料請(qǐng)關(guān)注服務(wù)器之家其它相關(guān)文章!

延伸 · 閱讀

精彩推薦
主站蜘蛛池模板: 女人zooxx禽交| 香蕉久久夜色精品国产小优 | 亚洲天堂v | 91会员 | 国产成人精品视频一区 | 亚洲性色永久网址 | 三级黄色片在线观看 | 色姑娘色综合 | 亚洲欧美久久久久久久久久爽网站 | 被老外玩爽的中国美女视频 | 亚洲精彩视频在线观看 | 好大好深视频 | 日本视频在线免费播放 | 无颜之月全集免费观看 | 午夜一区二区福利视频在线 | 秋霞宅宅236理论片 秋霞一级黄色片 | 三级黄色图片 | 18美女光胸光屁屁洗澡 | 成人aqq | 超级乱淫1| 国产精彩视频 | 齐天大性之七仙女欲春迅雷链接 | 99久久99久久久精品齐齐鬼色 | 亚洲国产香蕉视频欧美 | 强漂亮白丝女教师小说 | 日韩色在线观看 | 99国产精品久久久久久久... | 午夜dj免费视频观看社区 | 沉沦艳妇杨幂肉体小说 | 亚洲国产情侣一区二区三区 | 国产精品久久久久久久久久久威 | 加勒比久草 | 性色欲情网站IWWW九文堂 | 日本亚欧乱色视频在线观看 | kk4444在线影视播放 | 欧美在线视频一区二区 | 互换身体全集免费观看 | 国产99在线观看 | 日本一道本视频 | 好紧水好多 | 99爱在线精品视频免费观看9 |