所谓回调,就是客户程序C调用服务程序S中的某个函数SA,然后S又在某个时候反过来调用C中的某个函数CB,对于C来说,这个CB便叫做回调函数。例如Win32下的窗口过程函数就是一个典型的回调函数。一般说来,C不会自己调用CB,C提供CB的目的就是让S来调用它,而且是C不得不提供。由于S并不知道C提供的CB姓甚名谁,所以S会约定B的接口规范(函数原型),然后由C提前通过S的一个函数R告诉S自己将要使用CB函数,这个过程称为回调函数的注册,R称为注册函数。Web Service以及Java的RMI都用到回调机制,可以访问远程服务器程序。
一个通俗的例子。某天,我打电话向你请教问题,当然是个难题,^_^,你一时想不出解决方法,我又不能拿着电话在那里傻等,于是我们约定:等你想出办法后打手机通知我,这样,我就挂掉电话办其它事情去了。过了XX分钟,我的手机响了,你兴高采烈的说问题已经搞定,应该如此这般处理。故事到此结束。这个例子说明了“异步+回调”的编程模式。
熟悉MS-Windows和X Windows事件驱动设计模式的开发人员,通常是把一个方法的指针传递给事件源,当某一事件发生时来调用这个方法(也称为“回调”)。Java的面向对象的模型目前不支持方法指针,似乎不能使用这种方便的机制。
Java支持interface,通过interface可以实现相同的回调。其诀窍就在于定义一个简单的interface,申明一个被希望回调的方法。
例如,假定当某一事件发生时会得到通知,我们可以定义一个interface:
- public interface InterestingEvent {
-
- public void interestingEvent();
- }
这样我们就有了任何一个实现了这个接口类对象的手柄grip。
当一事件发生时,需要通知实现
InterestingEvent 接口的对象,并调用
interestingEvent() 方法。
- class EventNotifier {
- private InterestingEvent ie;
- private boolean somethingHappened;
- public EventNotifier(InterestingEvent event) {
- ie = event;
- somethingHappened = false;
- }
- public void doWork() {
- if (somethingHappened) {
-
- ie.interestingEvent();
- }
- }
- }
在这个例子中,用
somethingHappened 来标志事件是否发生。
希望接收事件通知的类必须要实现
InterestingEvent 接口,而且要把自己的引用传递给事件的通知者。
- public class CallMe implements InterestingEvent {
- private EventNotifier en;
- public CallMe() {
-
- en = new EventNotifier(this);
- }
-
- public void interestingEvent() {
-
- }
- }
以上是通过一个非常简单的例子来说明Java中的回调的实现。
当然,也可以在事件管理或事件通知者类中,通过注册的方式来注册多个对此事件感兴趣的对象。
1. 定义一个接口InterestingEvent ,回调方法nterestingEvent(String event) 简单接收一个String 参数。
- interface InterestingEvent {
- public void interestingEvent(String event);
- }
2. 实现InterestingEvent接口,事件处理类
- class CallMe implements InterestingEvent {
- private String name;
- public CallMe(String name){
- this.name = name;
- }
- public void interestingEvent(String event) {
- System.out.println(name + ":[" +event + "] happened");
- }
- }
3. 事件管理者,或事件通知者
- class EventNotifier {
- private List<CallMe> callMes = new ArrayList<CallMe>();
-
- public void regist(CallMe callMe){
- callMes.add(callMe);
- }
-
- public void doWork(){
- for(CallMe callMe: callMes) {
- callMe.interestingEvent("sample event");
- }
- }
- }
4. 测试
- public class CallMeTest {
- public static void main(String[] args) {
- EventNotifier ren = new EventNotifier();
- CallMe a = new CallMe("CallMe A");
- CallMe b = new CallMe("CallMe B");
-
- ren.regist(a);
- ren.regist(b);
-
-
- ren.doWork();
- }
- }