深入浅析Android接口回调机制
在使用接口回调的时候发现了一个经常犯的错误,就是回调函数里面的实现有可能是用多线程或者是异步任务去做的,这就会导致我们期望函数回调完毕去返回一个主函数的结果,实际发现是行不通的,因为如果回调是多线程的话你是无法和主函数同步的,也就是返回的数据是错误的,这是非常隐秘的一个错误。那有什么好的方法去实现数据的线性传递呢?先介绍下回调机制原理。
回调函数
回调函数就是一个通过函数指针调用的函数。如果你把函数的指针(地址)作为参数传递给另一个函数,当这个指针被用为调用它所指向的函数时,我们就说这是回调函数。回调函数不是由该函数的实现方直接调用,而是在特定的事件或条件发生时由另外的一方调用的,用于对该事件或条件进行响应。
开发中,接口回调是我们经常用到的。
接口回调的意思即,注册之后并不立马执行,而在某个时机触发执行。
举个例子:
a有一个问题不会,他去问b,b暂时解决不出来,b说,等我(b)解决了再告诉你(a)此时a可以继续先做别的事情。
那么就只有当b解决完问题后告诉a问题解决了,a才可以能解决这个问题。
代码中比如最常用的:
一个activity中给按钮一个接口回调方法,只有用户点击了这个按钮,告诉按钮被点击了,才会执行按钮接口回调的方法
button btn = new button(this); btn.setonclicklistener(new view.onclicklistener() { @override public void onclick(view view) { } });
那么下面通过一个demo理解接口回调:
主线程开启一个异步任务,当异步任务接收到数据,则把数据用textview显示出来
1、首先 我们需要定义一个接口,定义一个方法,参数为一个字符串:
package com.xqx.interfacedemo; public interface changetitle { void onchangetitle(string title); }
2、写一个异步任务,把接口作为构造方法参数,在doinbackground()方法中判断如果有数据,则接口回调
package com.xqx.interfacedemo; import android.content.context; import android.os.asynctask; public class mytask extends asynctask<string,void,string>{ private changetitle changetitle; public mytask(changetitle changetitle) { this.changetitle = changetitle; } @override protected string doinbackground(string... strings) { if (strings[0]!=null){ changetitle.onchangetitle(strings[0]); } return null; } }
3、主activity,给异步任务参数传this,即 接口回调方法在此类中执行,那么就需要实现changetitle接口,重写接口中
onchangetitle 方法
package com.xqx.interfacedemo; import android.app.activity; import android.os.bundle; import android.view.view; import android.widget.textview; public class mainactivity extends activity implements changetitle { private textview textview; @override public void oncreate(bundle savedinstancestate) { super.oncreate(savedinstancestate); setcontentview(r.layout.main); textview = (textview) findviewbyid(r.id.textview); new mytask(this).execute("我是标题"); } // 重写接口方法,执行相应操作 @override public void onchangetitle(string title) { textview.settext(title); } }
以上内容就是本文给大家分享的android接口回调机制,感谢大家对网站的关注,有你们的关注我们会做的更好,谢谢!
下一篇: 用 PHP5 轻松解析 XML