Java回调函数详解
为了了解什么是回调函数,在网上查阅了如下资料,整理如下:
资料一:
首先说说什么叫回调函数?
在WINDOWS中,程序员想让系统DLL调用自己编写的一个方法,于是利用DLL当中回调函数(CALLBACK)的接口来编写程序,使它调用,这个就 称为回调。在调用接口时,需要严格的按照定义的参数和方法调用,并且需要处理函数的异步,否则会导致程序的崩溃。
这样的解释似乎还是比较难懂,这里举个简 单的例子:
程序员A写了一段程序(程序a),其中预留有回调函数接口,并封装好了该程序。程序员B要让a调用自己的程序b中的一个方法,于是,他通过a中的接口回调自己b中的方法。目的达到。
在C/C++中,要用回调函数,被掉函数需要告诉调用者自己的指针地址,但在JAVA中没有指针,怎么办?我们可以通过接口(interface)来实现定义回调函数。
假设我是程序员A,以下是我的程序a:
- public class Caller {
- public MyCallInterface mc;
- public void setCallfuc(MyCallInterface mc){
- this.mc= mc;
- }
- public void call(){
- this.mc.method();
- }
- }
public class Caller { public MyCallInterface mc; public void setCallfuc(MyCallInterface mc){ this.mc= mc; } public void call(){ this.mc.method(); } }
我还需要定义一个接口,以便程序员B根据我的定义编写程序实现接口。
- public interface MyCallInterface {
- public void method();
- }
public interface MyCallInterface { public void method(); }
于是,程序员B只需要实现这个接口就能达到回调的目的了:
- public class B implements MyCallInterface {
- public void method() {
- System.out.println("谁来调用我啊。好寂寞!!");
- }
- public static void main(String args[]) {
- Caller call = new Caller();
- call.setCallfuc(new B());
- call.call();
- }
- }
public class B implements MyCallInterface { public void method() { System.out.println("谁来调用我啊。好寂寞!!"); } public static void main(String args[]) { Caller call = new Caller(); call.setCallfuc(new B()); call.call(); } }
资料二:
所谓回调,就是客户程序C调用服务程序S中的某个方法a,然后S又在某个时候反过来调用C中的某个方法b,对于C来说,这个b便叫做回调函数。
一般说来,C不会自己调用b,C提供b的目的就是让S来调用它,而且是C不得不提供。由于S并不知道C提供的b叫甚名谁,所以S会约定b的接口规范(函数原型),然后由C提前通过S的一个函数r告诉S自己将要使用b函数,这个过程称为回调函数的注册,r称为注册函数。
下面举个通俗的例子:
某天,我打电话向你请教问题,当然是个难题,:),你一时想不出解决方法,我又不能拿着电话在那里傻等,于是我们约定:等你想出办法后打手机通知我,这样,我就挂掉电话办其它事情去了。过了XX分钟,我的手机响了,你兴高采烈的说问题已经搞定,应该如此这般处理。故事到此结束。
这个例子说明了“异步+回调”的编程模式。其中,你后来打手机告诉我结果便是一个“回调”过程;我的手机号码必须在以前告诉你,这便是注册回调函数;我的手机号码应该有效并且手机能够接收到你的呼叫,这是回调函数必须符合接口规范。
如果你还不太清楚看看这段描述合和代码:
声明一个接口,另外一个类有方法里面有个参数以是这个接口类型的,而后在另外类中实现这个接口(java中多用的是匿名内部类),而且以这个匿名的类生成的对象为参数传到上面提到类中,而后实现回调.......这种用法可以参考java里面常用到的数据库操作所用到的几个接口.....
- //声明一个接口
- public interface ICallBack {
- void postExec();
- }
- //另外一个类有方法里面有个参数以是这个接口类型的
- public class FooBar {
- private ICallBack callBack;
- public void setCallBack(ICallBack callBack) {
- this.callBack = callBack;
- }
- public void doSth() {
- callBack.postExec();
- }
- }
- ----------------------------------------
- 回调的实现
- public class Test {
- public static void main(String[] args) {
- FooBar foo = new FooBar();
- foo.setCallBack(new ICallBack() {
- public void postExec() {
- System.out.println("method executed.");
- }
- });
- foo.doSth();//调用函数
- }
- }
//声明一个接口 public interface ICallBack { void postExec(); } //另外一个类有方法里面有个参数以是这个接口类型的 public class FooBar { private ICallBack callBack; public void setCallBack(ICallBack callBack) { this.callBack = callBack; } public void doSth() { callBack.postExec(); } } ---------------------------------------- 回调的实现 public class Test { public static void main(String[] args) { FooBar foo = new FooBar(); foo.setCallBack(new ICallBack() { public void postExec() { System.out.println("method executed."); } }); foo.doSth();//调用函数 } }
------------------------------------------------------------------------------------------------------------------------------------------------------------
资料三:一个利用回调函数的经典例子
下面使用java回调函数来实现一个测试函数运行时间的工具类:
如果我们要测试一个类的方法的执行时间,通常我们会这样做:
java 代码
- public class TestObject {
- /**
- * 一个用来被测试的方法,进行了一个比较耗时的循环
- */
- public static void testMethod(){
- for ( int i= 0 ; i< 100000000 ; i++){
- }
- }
- /**
- * 一个简单的测试方法执行时间的方法
- */
- public void testTime(){
- long begin = System.currentTimeMillis(); //测试起始时间
- testMethod(); //测试方法
- long end = System.currentTimeMillis(); //测试结束时间
- System.out.println("[use time]:" + (end - begin)); //打印使用时间
- }
- public static void main(String[] args) {
- TestObject test=new TestObject();
- test.testTime();
- }
- }
public class TestObject { /** * 一个用来被测试的方法,进行了一个比较耗时的循环 */ public static void testMethod(){ for ( int i= 0 ; i< 100000000 ; i++){ } } /** * 一个简单的测试方法执行时间的方法 */ public void testTime(){ long begin = System.currentTimeMillis(); //测试起始时间 testMethod(); //测试方法 long end = System.currentTimeMillis(); //测试结束时间 System.out.println("[use time]:" + (end - begin)); //打印使用时间 } public static void main(String[] args) { TestObject test=new TestObject(); test.testTime(); } }
大家看到了testTime()方法,就只有"//测试方法"是需要改变的,下面我们来做一个函数实现相同功能但更灵活:
首先定一个回调接口:
java 代码
- public interface CallBack {
- //执行回调操作的方法
- void execute();
- }
- 然后再写一个工具类:
- java 代码
- public class Tools {
- /**
- * 测试函数使用时间,通过定义CallBack接口的execute方法
- * @param callBack
- */
- public void testTime(CallBack callBack) {
- long begin = System.currentTimeMillis(); //测试起始时间
- callBack.execute(); ///进行回调操作
- long end = System.currentTimeMillis(); //测试结束时间
- System.out.println("[use time]:" + (end - begin)); //打印使用时间
- }
- public static void main(String[] args) {
- Tools tool = new Tools();
- tool.testTime(new CallBack(){
- //定义execute方法
- public void execute(){
- //这里可以加放一个或多个要测试运行时间的方法
- TestObject.testMethod();
- }
- });
- }
- }
public interface CallBack { //执行回调操作的方法 void execute(); } 然后再写一个工具类: java 代码 public class Tools { /** * 测试函数使用时间,通过定义CallBack接口的execute方法 * @param callBack */ public void testTime(CallBack callBack) { long begin = System.currentTimeMillis(); //测试起始时间 callBack.execute(); ///进行回调操作 long end = System.currentTimeMillis(); //测试结束时间 System.out.println("[use time]:" + (end - begin)); //打印使用时间 } public static void main(String[] args) { Tools tool = new Tools(); tool.testTime(new CallBack(){ //定义execute方法 public void execute(){ //这里可以加放一个或多个要测试运行时间的方法 TestObject.testMethod(); } }); } }
大家看到,testTime()传入定义callback接口的execute()方法就可以实现回调功能