回調(diào)概念:

  軟件模塊之間總是存在著一定的接口,從調(diào)用方式上,可以把他們分為三類:同步調(diào)用、回調(diào)和異步調(diào)用。同步調(diào)用是一種阻塞式調(diào)用,調(diào)用方要等待對(duì)方執(zhí)行完畢 才返回,它是一種單向調(diào)用;回調(diào)是一種雙向調(diào)用模式,也就是說(shuō),被調(diào)用方在接口被調(diào)用時(shí)也會(huì)調(diào)用對(duì)方的接口;異步調(diào)用是一種類似消息或事件的機(jī)制,不過(guò)它 的調(diào)用方向剛好相反,接口的服務(wù)在收到某種訊息或發(fā)生某種事件時(shí),會(huì)主動(dòng)通知客戶方(即調(diào)用客戶方的接口)。回調(diào)和異步調(diào)用的關(guān)系非常緊密,通常我們使用 回調(diào)來(lái)實(shí)現(xiàn)異步消息的注冊(cè),通過(guò)異步調(diào)用來(lái)實(shí)現(xiàn)消息的通知。同步調(diào)用是三者當(dāng)中最簡(jiǎn)單的,而回調(diào)又常常是異步調(diào)用的基礎(chǔ)。

  Java(Java教程 Java培訓(xùn) 實(shí)現(xiàn)回調(diào):

  在 Java 支持方法指針之前,Java 接口不能提供一種實(shí)現(xiàn)回調(diào)的好方法。如果您習(xí)慣于傳遞在事件驅(qū)動(dòng)編程模型中調(diào)用的函數(shù)指針,則您會(huì)喜歡本技巧。 熟悉 MS-Windows 和 X Window System 事件驅(qū)動(dòng)編程模型的開發(fā)人員,習(xí)慣于傳遞在某種事件發(fā)生時(shí)調(diào)用(即“回調(diào)”)的函數(shù)指針。Java 的面向?qū)ο竽P湍壳安⒉恢С址椒ㄖ羔槪@樣似乎就不可能使用這種很好的機(jī)制。但我們并不是一點(diǎn)辦法都沒有!

  Java 的接口支持提供了一種獲得回調(diào)的等價(jià)功能的機(jī)制。其技巧就是:定義一個(gè)簡(jiǎn)單接口,并在該接口中聲明我們要調(diào)用的方法。

  例如,假定我們希望在某個(gè)事件發(fā)生時(shí)得到通知。我們可以定義一個(gè)接口:

  public interface InterestingEvent

  { // 這僅是一個(gè)常規(guī)方法。因此如果需要,

  // 它可有返回值,也可接收參數(shù)。

  public void interestingEvent ();

  }

  這使得我們可以控制實(shí)現(xiàn)該接口的類的任何對(duì)象。因此,我們不必關(guān)心任何外部類型信息。與在將 C++ 代碼用于 Motif 時(shí)使用窗口小部件的數(shù)據(jù)域來(lái)容納對(duì)象指針的難以控制的 C 函數(shù)相比,這種方法要好得多。

  發(fā)出事件信號(hào)的類必須等待實(shí)現(xiàn)了 InterestingEvent 接口的對(duì)象,并在適當(dāng)時(shí)候調(diào)用 interestingEvent() 方法。

  public class EventNotifier

  { private InterestingEvent ie;

  private boolean somethingHappened;

  public EventNotifier (InterestingEvent event)

  { // 保存事件對(duì)象以備后用。

  ie = event;

  // 還沒有要報(bào)告的事件。

  somethingHappened = false;

  } //...

  public void doWork ()

  { // 檢查在別處設(shè)置的謂詞。

  if (somethingHappened)

  { // 通過(guò)調(diào)用接口的這個(gè)方法發(fā)出事件信號(hào)。

  ie.interestingEvent ();

  } //...

  }

  // ...

  }

  在上例中,我使用 somethingHappened 謂詞來(lái)跟蹤是否應(yīng)觸發(fā)事件。在許多情況下,調(diào)用此方法足以保證向 interestingEvent() 發(fā)出信號(hào)。

  希望接收事件通知的代碼必須實(shí)現(xiàn) InterestingEvent 接口,并將自身引用傳遞給事件通知程序。

  public class CallMe implements InterestingEvent

  { private EventNotifier en;

  public CallMe ()

  { // 創(chuàng)建事件通知程序,并將自身引用傳遞給它。

  en = new EventNotifier (this);

  }

  // 為事件定義實(shí)際的處理程序。

  public void interestingEvent ()

  { // 噢!必定發(fā)生了感興趣的事件!

  // 執(zhí)行某些操作 ...

  }

  //...

  }

  JAVA的CALLBACK通過(guò)接口來(lái)實(shí)現(xiàn)。

  例: 1.class A,class B

  2.class A實(shí)現(xiàn)接口operate

  3.class B擁有一個(gè)參數(shù)為operate接口類型的函數(shù)test(operate o)

  4.class A運(yùn)行時(shí)調(diào)用class B中test函數(shù),以自身傳入?yún)?shù)

  5.class B已取得A,就可以隨時(shí)回調(diào)A所實(shí)現(xiàn)的operate接口中的方法

  public interface BoomWTC

  { //獲得拉登的決定

  public benLaDengDecide();

  // 執(zhí)行轟炸世貿(mào)

  public void boom();

  }

  public class At$911 implements BoomWTC

  { private boolean decide;

  private TerroristAttack ta;

  public At$911(){

  Date now=new Date();

  SimpleDateFormat myFmt1=new SimpleDateFormat("yy/MM/dd HH:mm");

  this.dicede= myFmt.format(dt).equals("01/09/11 09:44");

  this.ta=new TerroristAttack();

  }

  //獲得拉登的決定

  public boolean benLaDengDecide(){

  return decide;

  }

  // 執(zhí)行轟炸世貿(mào)

  public void boom(){

  ta.attack(new At$911);

  } }

  public class TerroristAttack{

  public TerroristAttack(){

  }

  public attack(BoomWTC bmw){

  if(bmw.benLaDengDecide()){

  //let's go.........

  } }

  }