深入浅析Android接口回调机制
在使用接口回调的时候发现了一个经常犯的错误,就是回调函数里面的实现有可能是用多线程或者是异步任务去做的,这就会导致我们期望函数回调完毕去返回一个主函数的结果,实际发现是行不通的,因为如果回调是多线程的话你是无法和主函数同步的,也就是返回的数据是错误的,这是非常隐秘的一个错误。那有什么好的方法去实现数据的线性传递呢?先介绍下回调机制原理。
回调函数
回调函数就是一个通过函数指针调用的函数。如果你把函数的指针(地址)作为参数传递给另一个函数,当这个指针被用为调用它所指向的函数时,我们就说这是回调函数。回调函数不是由该函数的实现方直接调用,而是在特定的事件或条件发生时由另外的一方调用的,用于对该事件或条件进行响应。
开发中,接口回调是我们经常用到的。
接口回调的意思即,注册之后并不立马执行,而在某个时机触发执行。
举个例子:
A有一个问题不会,他去问B,B暂时解决不出来,B说,等我(B)解决了再告诉你(A)此时A可以继续先做别的事情。
那么就只有当B解决完问题后告诉A问题解决了,A才可以能解决这个问题。
代码中比如最常用的:
一个Activity中给按钮一个接口回调方法,只有用户点击了这个按钮,告诉按钮被点击了,才会执行按钮接口回调的方法
Buttonbtn=newButton(this); btn.setOnClickListener(newView.OnClickListener(){ @Override publicvoidonClick(Viewview){ } });
那么下面通过一个Demo理解接口回调:
主线程开启一个异步任务,当异步任务接收到数据,则把数据用TextView显示出来
1、首先我们需要定义一个接口,定义一个方法,参数为一个字符串:
packagecom.xqx.InterfaceDemo; publicinterfaceChangeTitle{ voidonChangeTitle(Stringtitle); }
2、写一个异步任务,把接口作为构造方法参数,在doInBackground()方法中判断如果有数据,则接口回调
packagecom.xqx.InterfaceDemo; importandroid.content.Context; importandroid.os.AsyncTask; publicclassMyTaskextendsAsyncTask<String,Void,String>{ privateChangeTitlechangeTitle; publicMyTask(ChangeTitlechangeTitle){ this.changeTitle=changeTitle; } @Override protectedStringdoInBackground(String...strings){ if(strings[0]!=null){ changeTitle.onChangeTitle(strings[0]); } returnnull; } }
3、主Activity,给异步任务参数传this,即接口回调方法在此类中执行,那么就需要实现ChangeTitle接口,重写接口中
onChangeTitle方法
packagecom.xqx.InterfaceDemo; importandroid.app.Activity; importandroid.os.Bundle; importandroid.view.View; importandroid.widget.TextView; publicclassMainActivityextendsActivityimplementsChangeTitle{ privateTextViewtextView; @Override publicvoidonCreate(BundlesavedInstanceState){ super.onCreate(savedInstanceState); setContentView(R.layout.main); textView=(TextView)findViewById(R.id.textView); newMyTask(this).execute("我是标题"); } //重写接口方法,执行相应操作 @Override publicvoidonChangeTitle(Stringtitle){ textView.setText(title); } }
以上内容就是本文给大家分享的Android接口回调机制,感谢大家对毛票票网站的关注,有你们的关注我们会做的更好,谢谢!