Android 的回调事件详解
看见网上一些回调的解释都很复杂的,特别基于Android的自定义回调,感觉一头雾水,于是乎,我也写了这篇基于我对回调的解释。
先来看一个简单的例子:
有两个类ClassA,和ClassB,ClassA调用ClassB里面的方法,
publicclassClassB{ publicvoidmethod_from_classB(){ for(inti=0;i<10;i++) System.out.print("..."+i); } } publicclassClassA{ publicstaticvoidmain(Stringargs[]){ ClassBclassB=newClassB(); classB.method_from_classB(); } }
输出:
...0...1...2...3...4...5...6...7...8...9
卧槽,哪个傻逼写的博文,侮辱我的智商不是吗,嘻嘻,是为了做比较,接下来看看利用回调,ClassA是怎么调用ClassB中的方法的,注意是回调:
让ClassB实现ClassA定义的接口
publicclassClassBimplementsClassA.ClassAInterface{ publicClassB(){ newClassA().RegisterInterface(this); System.out.println("...ClassB..."+this); } @Override publicvoidmethod_from_interface(){ for(inti=0;i<10;i++) System.out.print("..."+i); } /*publicvoidmethod_from_classB(){ for(inti=0;i<10;i++) System.out.print("..."+i); }*/ }
ClassA里面定义接口和抽象方法:
publicclassClassA{ publicstaticClassAInterfaceclassAInterface; publicinterfaceClassAInterface{ publicvoidmethod_from_interface(); } publicvoidRegisterInterface(ClassAInterfacea_interface){ this.classAInterface=a_interface; System.out.println("...a_interface..."+a_interface); } publicstaticvoidmain(Stringargs[]){ ClassBclassB=newClassB();//标记@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中的方法):
publicvoidRegisterInterface(ClassAInterfacea_interface){ this.classAInterface=a_interface; System.out.println("...a_interface..."+a_interface); }
输出:
...a_interface...ClassB@3ddb8962
第二个:
publicClassB(){ newClassA().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();
这是不是跟最上面到的代码:
ClassBclassB=newClassB(); classB.method_from_classB();
一样呢,这也是为什么我最开始要举这个例子的原因!!!
相信看到这里应该理解了接口的回调是怎么回事了吧。
但有一点又糊涂了,为什么要接口回调这么麻烦的,最上面的在ClassA里面执行:
ClassBclassB=newClassB(); classB.method_from_classB();
不是照样可以ClassA调用ClassB里面的方法。。。。但要是ClassA要调用ClassC,ClassD...,里面的方法呢,是不是还要改变ClassA里面的代码,实例化ClassC,ClassD...的对象,显然是不好的,要是使用接口那就不用改变ClassA里面的代码了,任何类只要实现ClassA里面的接口就可以.
解释一下标记@1:
上面那段话好像跟标记@1违背了,在ClassA里面确实也需要实例化ClassB对象。
因为要【利用】初始化的时候执行构造方法里面的代码:
publicClassB(){ //相当于回调事件的注册,初学者出现回调空指针很有可能这边忘记‘注册'了 newClassA().RegisterInterface(this); }
将this传递给ClassA,作用也就是上面利用日志分析的作用。
但再Android开发中救你不必这样了,
可以在Activity的初始化时执行:
@Override protectedvoidonCreate(BundlesavedInstanceState){ //相当于回调事件的注册,初学者出现回调空指针很有可能这边忘记‘注册'了 newClassA().RegisterInterface(this); }
在Android开发中ClassA里面的mian()函数可以用事件来代替,触发:
如:
button.setOnClickListener(newOnClickListener(){ @Override publicvoidonClick(Viewv){ //TODO自动生成的方法存根 if(classAInterface!=null){ classAInterface.method_from_interface(); } });