java回调机制及其实现

1. 什么是回调函数

回调函数,顾名思义,用于回调的函数。回调函数只是一个功能片段,由用户按照回调函数调用约定来实现的一个函数。回调函数是一个工作流的一部分,由工作流来决定函数的调用(回调)时机。回调函数包含下面几个特性:

(1)属于工作流的一个部分;

(2)必须按照工作流指定的调用约定来申明(定义);

(3)他的调用时机由工作流决定,回调函数的实现者不能直接调用回调函数来实现工作流的功能;

2. 回调机制

回调机制是一种常见的设计模式,它把工作流内的某个供,按照约定的接口暴露给外部使用者,为外部使用者提供数据,或要求外部使用者提供数据。

3.java的回调机制

软件模块之间总是存在这一定的接口,从调用方式上,可以把他们分为三类:同步调用、回调、异步调用。

同步调用:一种阻塞式调用,调用方要等待对方执行完毕才能返回,它是一种单向调用;

回调:一种双向调用模式,也就是说,被调用方在接口被调用时也会调用对方的接口

异步调用:一种类似消息或事件的机制,不过它的调用方向刚好相反,接口的服务在收到某种讯息或发生某种事件时,会主动通知客户方(即调用客户方的接口)。

回调和异步调用的关系非常紧密:使用回调来实现异步消息的注册,通过异步调用来实现消息的通知。

4.回调的java例子

例1:

        public static void main(String[] args) {
                FooBar fooBar = new FooBar();
                fooBar.setCallback(new ICallback() {
                        public void postExec() {
                                System.out.println("在CallbackExample类中实现但不能被CallbackExample的对象调用,而由FooBar对象调用");
                        }
                });
        }
}

interface ICallback {
        //需要回调的方法
        public void postExec();
}

class FooBar {
        private ICallback callback;
        public void setCallback(ICallback callback) {
                this.callback = callback;
                doSth();
        }
        
        public void doSth() {
                callback.postExec();
        }
}

例2:

        public static void main(String[] args) {
                new Worker().doWork();
        }
}

interface FetchCallback {
        void onData(Data data);
        void onError(Throwable cause);
} 

interface Fetcher {
        void fetchData(FetchCallback callback);
}

class Worker {
        Data data = new Data();
        public void doWork() {
                Fetcher fetcher = new Fetcher() {
                        public void fetchData(FetchCallback callback) {
                                System.out.println("start fetch data...");
                                callback.onData(data);
                        }
                };
                fetcher.fetchData(new FetchCallback() {
                        public void onData(Data data) { // call if data is fetched without error
                                System.out.println("Data received: " + data);
                        }
                        
                        public void onError(Throwable cause) { //call if error is received during fetch
                                System.err.println("An error accour: " + cause.getMessage());
                        }
                });
        }
}

class Data {
        
}