一、前言
本周有位入行开发不久的朋友问我回调究竟是个什么概念,在网上看了很多的回调函数解释,但是越看越乱。虽然回调函数这个梗已经不新鲜了,这里还是用书面的形式记录下。
如果有了解的,就无需再看。
二、概念
概念上,这里引用百度百科的解释,如下:
回调函数就是一个通过函数指针调用的函数。如果你把函数的指针(地址)作为参数传递给另一个函数,当这个指针被用来调用其所指向的函数时,我们就说这是回调函数。回调函数不是由该函数的实现方直接调用,而是在特定的事件或条件发生时由另外的一方调用的,用于对该事件或条件进行响应。
百度百科的定义就是上面这样的,它提到了函数指针这个概念,函数指针一般是C/C++的专有名词。Java中也是有这个概念的,只是我们把它叫做引用,淡化了函数指针的概念,使用也更加简单。
这个概念的定义,看上去是有点绕,并且不容易理解。网上有很多的对回调函数的说明,比如,A类调用B类的方法B1,B类又调用A类的方法A1之类。这样的说法,其实看的也很迷茫。
因此,会在下文中对回调函数做一个层次分明的解释。
三、元素
根据概念,我们知道一个回调函数的调用流程是需要以下三个元素,并分别给它们一个名字:
1、回调函数本身——回调函数;
2、回调函数作为参数传入的函数——中间函数;
3、调用者(调用函数)——调用函数。
根据以上的命名,回调函数的流程就是:
(1)将回调函数的引用,传入到中间函数(这个也可以称之为回调函数的注册/订阅)。
(2)调用函数调用中间函数,触发回调函数的事件。
纯文字说明可能没有感觉,这里我们引入一个很简单的例子,来描述这一流程:
首先,你需要一个回调函数:
package com.callback; public class Callback { public void call(){ System.out.println("我是一个回调函数,当我被打印出来的时候,说明回调函数被触发了"); } }
其次,再来一个中间函数:
package com.callback; public class Middle { //参数是回调函数所在类的引用 public void mid(Callback callback){ //触发回调函数 callback.call(); } }
最后,是调用函数:
package com.callback; public class Main { /** * @param args */ public static void main(String[] args) { Middle m=new Middle(); //回调函数注册,将Callback的引用传入中间函数 m.mid(new Callback()); } }
运行一下,你就可以看到打印结果:
我是一个回调函数,当我被打印出来的时候,说明回调函数被触发了
以上是一个简单例子,在该例子中,用最简方式模仿了回调函数的调用过程。但日常中我们是不会或者很少这么用的,因为该例子没有很好的表现出回调函数的作用。
回调函数有什么作用?在百度百科上有一段对它们意义的说明:
1、因为可以把调用者与被调用者分开,所以调用者不关心谁是被调用者。它只需知道存在一个具有特定原型和限制条件的被调用函数。
2、回调可用于通知机制。
3、这一设计允许了底层代码调用在高层定义的子程序。
简单的说,可以用于解耦、通知以及其他。
为了完整说明回调函数的作用,我们再原来的例子上,做个扩展,引入接口。扩展的还是上面的例子,引入接口:
新增接口:
package com.callback; public interface CallbackInterface { public void call(); }
原有的回调函数,实现上面的接口:
package com.callback; public class Callback implements CallbackInterface{ public void call() { // TODO Auto-generated method stub System.out.println("我是一个回调函数,当我被打印出来的时候,说明回调函数被触发了"); } }
中间函数的参数,做以下修改(修改为接口):
package com.callback; public class Middle { //参数是回调函数所在类的引用 public void mid(CallbackInterface callback){ //触发回调函数 callback.call(); } }
最后是调用函数:
package com.callback; public class Main { /** * @param args */ public static void main(String[] args) { Middle m=new Middle(); //回调函数注册 CallbackInterface ifsImpl=new Callback(); m.mid(ifsImpl); } }
对比两个例子,在后面这个例子中,引入接口,实现了:调用函数、中间函数与回调函数的解耦。底层函数对高层函数的调用。通知机制。
四、回调函数作为参数传入的方法
这是一个小细节,设计人员可以仿照依赖注入的三种方式,来实现回调函数的参数传入。
1、构造函数中传入;
2、用set方式传入;
3、直接将函数(接口)作为参数传入;
例子中就是第三种方式。
到此,回调函数的介绍就基本结束了。但是,上面的两个例子和我们实际接触的还是有那么一些差距。
在看完上面的例子后,对回调有了比较深的认识。现在,已经可以理论联系实际了。
也举一个日常用的很多的场景:Android中的按钮点击事件(这也是应用最广泛的一个回调函数)。
五、Andriod按钮点击事件的模拟对比
1、新开一个Android工程,我们做一个真实的按钮点击事件,如下:
//原始 Button btn=(Button)findViewById(R.id.btn); btn.setOnClickListener(new OnClickListener() { @Override public void onClick(View arg0) { // TODO Auto-generated method stub System.out.println("按钮被点击了-原始"); } });
2、再根据前面我们自己的分析,做一个按钮点击的回调函数事件:
也首先是一个回调函数:
package com.example.callbackandroid; public interface MyOnClickListener { //回调函数 public void onClick(); }
中间函数:
package com.example.callbackandroid; public class MyButton { private MyOnClickListener listener; //回调函数注册/订阅/登记 public void setOnclickListener(MyOnClickListener listener){ this.listener=listener; } public void doOnclick(){ listener.onClick(); } }
调用函数:
//模拟 MyButton mButton=new MyButton(); mButton.setOnclickListener(new MyOnClickListener() { @Override public void onClick() { // TODO Auto-generated method stub System.out.println("按钮被点击了-原始"); } }); mButton.doOnclick();
对比原始的android点击事件以及我们模拟的android点击事件,发现有一点区别:原始的android点击不需要没有调用doOnclick事件,没有调用函数??
实际上,原始的android点击事件是有调用函数的,只是不需要写在这里而已。如果有看过Android的onClick事件(事件分发)源码的朋友,会发现:在源码中,当android系统检测到该View的ACTION_UP的操作时,会调用performClick()这个函数,而这个函数的内容如下:
public boolean performClick() { sendAccessibilityEvent(AccessibilityEvent.TYPE_VIEW_CLICKED); if (mOnClickListener != null) { playSoundEffect(SoundEffectConstants.CLICK); mOnClickListener.onClick(this); return true; } return false; }
它的调用函数,是在这里实现的。只要mOnClickListener不为null(这个就是通过我们setOnClickListener来赋值的),那么onClick就会被调用。
因此,它和模拟点击原理其实是一样的。
到这里,回调函数的分析就告一段落了。
下一段落,主要是一个概念的整理。
六、回调函数与同步、异步
在真正使用回调函数的时,我们经常会接触到异步回调、异步调用、同步调用这些词。这里简单说下概念,以免混乱,作为结尾。
同步调用:
这个是日常使用最频繁的一种调用方法,这是一个阻塞调用,如你调用一个方法,等待这个方法返回,或者执行完毕。
异步调用:
这个在android中也使用频繁,非阻塞调用,如你调用一个方法,无需等这个方法返回,即往下执行其他操作。
异步回调:
这个是异步+回调的形式。