看见网上一些回调的解释都很复杂的,特别基于Android的自定义回调,感觉一头雾水,于是乎,我也写了这篇基于我对回调的解释。
先来看一个简单的例子:
有两个类 ClassA ,和 ClassB, ClassA调用ClassB里面的方法,
public class ClassB { public void method_from_classB(){ for(int i=0;i<10;i++) System.out.print("..."+i); } } public class ClassA { public static void main(String args[]){ ClassB classB = new ClassB(); classB.method_from_classB(); } }
输出:
...0...1...2...3...4...5...6...7...8...9
卧槽,哪个傻逼写的博文,侮辱我的智商不是吗,嘻嘻,是为了做比较,接下来看看利用回调, ClassA 是怎么调用 ClassB中的 方法的,注意是回调:
让ClassB 实现 ClassA定义的接口
public class ClassB implements ClassA.ClassAInterface{ public ClassB(){ new ClassA().RegisterInterface(this); System.out.println("...ClassB..."+this); } @Override public void method_from_interface() { for(int i=0;i<10;i++) System.out.print("..."+i); } /* public void method_from_classB(){ for(int i=0;i<10;i++) System.out.print("..."+i); }*/ }
ClassA里面定义接口和抽象方法:
public class ClassA { public static ClassAInterface classAInterface; public interface ClassAInterface{ public void method_from_interface(); } public void RegisterInterface(ClassAInterface a_interface){ this.classAInterface = a_interface; System.out.println("...a_interface..."+a_interface); } public static void main(String args[]){ ClassB classB = new ClassB();// 标记@1,最后面做解释 //classB.method_from_classB(); System.out.println("...classAInterface..."+classAInterface); if(classAInterface != null){ classAInterface.method_from_interface(); } } }
输出:
...0...1...2...3...4...5...6...7...8...9
整理下,也就是 我在ClassA里面定义了一个接口(interface),接口里面又定义了一个方法,但没有方法体,也就不做任何事情。
当 ClassA 执行到 mian() 函数时,就会调用接口的方法,但前面讲了,接口的方法没有实现具体的事情,它就会找到 ClassB 里面对应的 方法,来实现具体的事情。
呦呦呦,ClassA 的接口的方法是怎么找到 ClassB 的方法,难道会上天???
也就是下面分析这句代码是怎么上天的:
// 利用接口的回调实现 ClassB中 的方法的 具体事情
classAInterface.method_from_interface();
我在上面的代码中用 System.out.println 打印出了日志做分析:
第一个(ClassA中的方法):
public void RegisterInterface(ClassAInterface a_interface){ this.classAInterface = a_interface; System.out.println("...a_interface..."+a_interface); }
输出:
...a_interface...ClassB@3ddb8962
第二个:
public ClassB(){ new ClassA().RegisterInterface(this); System.out.println("...ClassB..."+this); }
输出:
...ClassB...ClassB@3ddb8962
第三个:
System.out.println("...classAInterface..."+classAInterface); if(classAInterface != null){ classAInterface.method_from_interface(); }
输出:
...classAInterface...ClassB@3ddb8962
看到这里是不是恍然大悟呢 ,输出都是 “ ClassB@3ddb8962 ” 也就是ClassB 对象的引用!!!
啊!接口只不过是将 ClassB 对象的引用 传到 ClassA中而已,那这句会上天的语句是不是很好解释了呢。
classAInterface.method_from_interface();
相当于 ClassB@3ddb8962.method_from_interface();
这是不是跟最上面到的代码:
ClassB classB = new ClassB(); classB.method_from_classB();
一样呢,这也是为什么我最开始要举这个例子的原因!!!
相信看到这里应该理解了接口的回调是怎么回事了吧。
但有一点又糊涂了,为什么 要接口回调这么麻烦的,最上面的在ClassA里面执行:
ClassB classB = new ClassB(); classB.method_from_classB();
不是照样可以 ClassA 调用 ClassB 里面的 方法。。。。但要是ClassA 要调用ClassC,ClassD ...,里面的方法呢,是不是还要改变ClassA里面的代码,实例化ClassC,ClassD ... 的对象,显然是不好的,要是使用接口那就不用改变ClassA 里面的代码了,任何类只要实现ClassA 里面的接口就可以.
解释一下 标记@1 :
上面那段话好像跟 标记@1 违背了,在 ClassA 里面确实也需要实例化 ClassB对象。
因为要 【利用】 初始化的时候执行构造方法里面的代码:
public ClassB(){ // 相当于回调事件的注册,初学者出现回调空指针很有可能这边忘记‘注册'了 new ClassA().RegisterInterface(this); }
将this 传递给 ClassA ,作用也就是 上面利用 日志分析的作用。
但再 Android 开发中救你不必这样了,
可以在 Activity 的初始化时执行:
@Override protected void onCreate(Bundle savedInstanceState) { // 相当于回调事件的注册,初学者出现回调空指针很有可能这边忘记‘注册'了 new ClassA().RegisterInterface(this); }
在Android 开发中 ClassA 里面的 mian() 函数可以用事件来代替,触发:
如:
button.setOnClickListener(new OnClickListener() { @Override public void onClick(View v) { // TODO 自动生成的方法存根 if(classAInterface != null){ classAInterface.method_from_interface(); } });
本文向大家介绍详解C#中委托,事件与回调函数讲解,包括了详解C#中委托,事件与回调函数讲解的使用技巧和注意事项,需要的朋友参考一下 .Net编程中最经常用的元素,事件必然是其中之一。无论在ASP.NET还是WINFrom开发中,窗体加载(Load),绘制(Paint),初始化(Init)等等。 “protected void Page_Load(object sender, EventArgs e
本文向大家介绍Android 回调详解及简单实例,包括了Android 回调详解及简单实例的使用技巧和注意事项,需要的朋友参考一下 Android 回调 前言: Android中的回调最经典的就是点击事件设置监听(一般通过switch(v.getId()))这里写个最基本的 view对外暴露了一个接口onClick 我们在用的时候实现具体方法写了处理时,当前的Actvity是没有对这个方法做
本文向大家介绍Android dataBinding与ListView及事件详解,包括了Android dataBinding与ListView及事件详解的使用技巧和注意事项,需要的朋友参考一下 今天来了解一下Android最新给我们带来的数据绑定框架——Data Binding Library。数据绑定框架给我们带来了更大的方便性,以前我们可能需要在Activity里写很多的findViewBy
update 动画开始播放后,每帧都会触发此回调。 Type Parameters Info Function animation 返回当前动画对象 var updates = 0; anime({ targets: '.update-demo .el', translateX: 270, delay: 1000, direction: 'alternate', loop:
注册事件回调函数,与swoole_server->on相同。 $server->on('eventName','function for call back') 的方式来实现回调事件的注册。 request 事件 $http_server->on('request', function(swoole_http_request $request, swoole_http_response $res
本文向大家介绍Android View 事件分发机制详解,包括了Android View 事件分发机制详解的使用技巧和注意事项,需要的朋友参考一下 Android开发,触控无处不在。对于一些 不咋看源码的同学来说,多少对这块都会有一些疑惑。View事件的分发机制,不仅在做业务需求中会碰到这些问题,在一些面试笔试题中也常有人问,可谓是老生常谈了。我以前也看过很多人写的这方面的文章,不是说的太啰嗦就是