欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Java回調(diào)方法詳解

 更新時(shí)間:2017年01月04日 10:48:52   作者:byhieg  
本篇文章主要介紹了Java回調(diào)方法的步驟、回調(diào)的例子、異步回調(diào)與同步回調(diào)、回調(diào)方法在通信中的應(yīng)用等。具有一定的參考價(jià)值,下面跟著小編一起來看下吧

回調(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)。

步驟如下:

  1. 在擁有數(shù)據(jù)的那個(gè)類里面寫一個(gè)回調(diào)的接口。-->這里就是B類中寫一個(gè)回調(diào)接口
  2. 回調(diào)方法接收一個(gè)參數(shù),這個(gè)參數(shù)就是要得到的數(shù)據(jù)
  3. 同樣是在這個(gè)類里寫一個(gè)注冊(cè)回調(diào)的方法。
  4. 在注冊(cè)回調(diào)方法,用接口的引用去調(diào)用回調(diào)接口,把B類的數(shù)據(jù)當(dāng)做參數(shù)傳入回調(diào)的方法中。
  5. 在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)文章

  • 使用Maven配置Spring的方法步驟

    使用Maven配置Spring的方法步驟

    這篇文章主要介紹了使用Maven配置Spring的方法步驟,小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧
    2019-04-04
  • java編程簡(jiǎn)單獲取圖片像素的方法

    java編程簡(jiǎn)單獲取圖片像素的方法

    這篇文章主要介紹了java編程簡(jiǎn)單獲取圖片像素的方法,涉及Java針對(duì)圖片的讀取與屬性獲取技巧,具有一定參考借鑒價(jià)值,需要的朋友可以參考下
    2015-11-11
  • java實(shí)現(xiàn)字符串匹配求兩個(gè)字符串的最大公共子串

    java實(shí)現(xiàn)字符串匹配求兩個(gè)字符串的最大公共子串

    這篇文章主要介紹了java實(shí)現(xiàn)求兩個(gè)字符串最大公共子串的方法,詳細(xì)的描述了兩個(gè)字符串的最大公共子串算法的實(shí)現(xiàn),需要的朋友可以參考下
    2016-10-10
  • Spring?Data?JPA系列QueryByExampleExecutor使用詳解

    Spring?Data?JPA系列QueryByExampleExecutor使用詳解

    這篇文章主要為大家介紹了Spring?Data?JPA系列QueryByExampleExecutor使用示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2022-09-09
  • java關(guān)于調(diào)用方法的匯總

    java關(guān)于調(diào)用方法的匯總

    本文小編給大家整理了在Java中關(guān)于靜態(tài)調(diào)用和動(dòng)態(tài)調(diào)用的方法匯總,值得大家學(xué)習(xí)和參考。
    2017-11-11
  • 淺談java中BigDecimal的equals與compareTo的區(qū)別

    淺談java中BigDecimal的equals與compareTo的區(qū)別

    下面小編就為大家?guī)硪黄獪\談java中BigDecimal的equals與compareTo的區(qū)別。小編覺得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧
    2016-11-11
  • 一文詳解Java中Map和Set接口的使用方法

    一文詳解Java中Map和Set接口的使用方法

    Map和set是一種專門用來進(jìn)行搜索的容器或者數(shù)據(jù)結(jié)構(gòu),其搜索的效率與其具體的實(shí)例化子類有關(guān),可能在查找時(shí)進(jìn)行一些插入和刪除的操作,即動(dòng)態(tài)查找,那上述兩種方式就不太適合了,本節(jié)介紹的Map和Set是一種適合動(dòng)態(tài)查找的集合容器,需要的朋友可以參考下
    2024-08-08
  • SpringBoot項(xiàng)目微信云托管入門部署實(shí)踐

    SpringBoot項(xiàng)目微信云托管入門部署實(shí)踐

    本文主要介紹了SpringBoot項(xiàng)目微信云托管入門部署實(shí)踐,文中通過示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2022-03-03
  • Mybatis插入時(shí)返回自增主鍵方式(selectKey和useGeneratedKeys)

    Mybatis插入時(shí)返回自增主鍵方式(selectKey和useGeneratedKeys)

    這篇文章主要介紹了Mybatis插入時(shí)返回自增主鍵方式(selectKey和useGeneratedKeys),具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2021-09-09
  • Java將字節(jié)轉(zhuǎn)換為十六進(jìn)制代碼分享

    Java將字節(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

最新評(píng)論