试想这么一个情景:A需要做某件事,让B帮忙做,B说我忙完手上的事情就去帮你做。但A总不能在哪干等着吧,因此,A就继续去做其他事情,让B做完之后通知A。
对于一个这样的业务场景,解决方案有两种,一种是上文我们介绍的通过Callable和Future来实现。另一种方案就是本文所介绍的回调机制。
为了方便理解,我们先说说什么叫调用?模块之间存在着一定的接口,供别人调用。而所谓的回调就是两个模块(对象)相互调用,包括同步调用和异步调用。但回调绝对不是为了解决调用时等待时间过长的问题(这是异步的工作),根据是否等待进一步分为同步和异步。回调函数的实现一般有三个特点:
特点1:类A需要实现我们定义的回调接口,方便B通知A。
特点2:类A需要包含一个类B的引用
特点3:类B中有一个参数为回调接口的函数
下面我们来看下异步回调的具体实现,我们以经典的打电话为例,有一天小王遇到一个很难的问题,问题是“1 + 1 = ?”,就打电话问小李,小李一下子也不知道,就跟小王说,等我办完手上的事情,就去想想答案,小王也不会傻傻的拿着电话去等小李的答案吧,于是小王就对小李说,我还要去逛街,你知道了答案就打我电话告诉我,于是挂了电话,自己办自己的事情,过了一个小时,小李打了小王的电话,告诉他答案是2。
首先我们需要定义一个回调接口,用于B完成A交代的任务后特告A我做完了。
/**
* 这是一个回调接口
*
*/
public interface CallBack {
/**
* 这个是小李知道答案时要调用的函数告诉小王,也就是回调函数
* @param result 是答案
*/
public void solve(String result);
}
声明了回调接口后,对于A,需要满足两个回调机制的特点,首先需要实现我们的回调接口CallBack,然后再声明一个B的引用,方便让B做事情。这里需要小王调用小李的方法。
/**
* 这个是小王
* 实现了一个回调接口CallBack,回调机制特点1
*/
public class Wang implements CallBack {
/**
* 小李对象的引用 相当回调机制特点2
*/
private Li li;
/**
* 小王的构造方法,持有小李的引用
* @param li
*/
public Wang(Li li){
this.li = li;
}
/**
* 小王通过这个方法去问小李的问题
* @param question 就是小王要问的问题,1 + 1 = ?
*/
public void askQuestion(final String question){
//这里用一个线程就是异步,
new Thread(new Runnable() {
@Override
public void run() {
/**
* 小王调用小李中的方法,在这里注册回调接口
* 体现的类A调用类B
*/
li.executeMessage(Wang.this, question);
}
}).start();
//小网问完问题挂掉电话就去干其他的事情
play();
}
public void play(){
System.out.println("我去忙了");
}
/**
* 小李知道答案后调用此方法告诉小王,就是所谓的小王的回调方法
*小李调用小王的这个方法
*/
@Override
public void solve(String result) {
System.out.println("小李告诉小王的答案是--->" + result);
}
}
接下来是小李,同样,小李也需要调用小王的方法这样双向才能满足回调。
/**
* 这个就是小李啦
*
*/
public class Li {
/**
* 相当于B类有参数为CallBack callBack的f(),回调机制特点3
* @param callBack
* @param question 小王问的问题
*/
public void executeMessage(CallBack callBack, String question){
System.out.println("小王问的问题--->" + question);
//模拟小李办自己的事情需要很长时间
for(int i=0; i<10000;i++){
}
/**
* 小李办完自己的事情之后想到了答案是2
*/
String result = "答案是2";
/**
* 于是就打电话告诉小王,调用小王中的方法
* 这就相当于B类反过来调用A的方法
*/
callBack.solve(result);
}
}
回调函数的实现是一个相互调用的过程,只要牢记我们提到的回调函数的三个特点,就能很容易的写出函调程序。