首页 > 代码库 > 浅谈“回调”在程序设计中的好处

浅谈“回调”在程序设计中的好处

1:回调还是返回(return)

在写代码的时候,我们经常碰到这样的场景:调用一个函数或者方法时需要返回多个值给上级调用者,如示例:

void methodA(){
   Wrap w = methodB(); 
   w.one; //use
   w.two;
}

Wrap methodB(){
    do something;
    return Wrap;
}

class Wrap{
   Type one;
   Type two;
}

上面是我刚开始写代码时候常用的方式,在多个类型的基础上再封装一个类,返回这个类类型给上层程序处理。在类似情景不多的情况下,采用这种处理方式 是可以接受的。慢慢的,当一个Project的代码越来越多,类似的情况出现频繁的时候,我就开始感到这种处理方式恶心的地方了。

1:为了返回几个不同类型的值,多出来了一个奇怪的类(还得为它想个名字,赋值(set),取值(get),多了n行代码)....

2:返回结果为null ? 为什么会出现null, 出现后怎么处理? 通过向上层抛出异常来处理?每一次遇到这种情况还得回去看原先写的代码,各种陷阱啊~~

更优雅的方式:

interface Callback{
    void onCall(Integer one,String two);
    void onFail(errorMsg);
}

void methodA(){
   methodB(new Callback(){
        void onCall(Integer one,String two){
              //取得返回值  
        }
        void onFail(errorMsg){
             //失败
       }
   }); 
}

void methodB(Callback call){
    if(success){ //正常
        call.onCall(one,two);//one和two为变量
    }else{ //有状况
        call.onFail(‘get fail‘);
    }
}

现在处理返回结果就好多了,代码的可读性也得到了增强,当然,如果你要求返回的结果非常多,还是要封装成一个类比较好点~

2:事件通知

我们先来看一段常用的JS代码

$(‘#id‘).click(function(){
   alert(‘click‘);
});

这段代码的工作流程大概就是浏览器捕获到输入设备(鼠标)的点击状态,然后执行click里面的匿名函数,完成整个回调的过程。

类似的情形我们还可以看到很多,尤其在UI的编程中,很多库几乎就是基于事件模型来编写的。下面我们来模拟下载文件进度的事件监听。

interface OnLoadListener{
   /**
    *@param progress 进度(百分比)
    */
   void onLoad(double progress);
}

void download(url,OnLoadListener listener){
    InputStream is = getStreamOfUrl(url); //
    double totalLen = is.ContentLength; //文件长度 
    int readLen = 0, hReadLen;
    while((readLen=is.read())!=-1){ //从网络上读取.
         hReadLen += readLen;
         listener.onLoad(hReadLen/totalLen); //把结果即时返回.
    }
}

//当我们在UI层调用时:
void show(){
  download("http://www.darcye.com/file",new OnLoadListener(){
       void onLoad(double progress){
             //显示进度
       }
  });
}

当然,这里只是简单的示例,目的在于说明回调在事件编程模型中的作用。实际上,还有异步,事件队列等许多问题需要进行考虑。

3:分离变化和不变的部分

我们知道,算法的步骤是一样的,只是数据源的类型不一致而已。因此我们可以想办法把不变的算法给分离出来。

相信了解Java的朋友都知道,在API中对对象的排序就是通过回调来解决这个问题的。也就是一种著名的设计模式-模板方法。让我们来看看它的实现过程吧。

我们从这个方法出发:

void java.util.Arrays.sort(T[] a, Comparator
   
    c)

深入源码很容易知道,排序的算法就是在这里实现的了,可以看到实现的是合并排序(merge sort)。排序根据(变化的部分)的关键就是这个接口Comparator,再深入代码就可以看到其会回调响应对象实现的接口方法compareTo()。