16、后台异步处理
程序员文章站
2022-03-11 10:38:06
...
大多时候,需要后台(异步)去执行一些费时操作。如网络数据获取、复杂的算法、图片处理等,当执行这些操作时,又不想失去用户响应(类似Windows有时候程序会卡死,失去鼠标键盘响应),所以需要放到用户界面之外的一个线程里去处理这些操作。
安卓中主要有两种实现方式,1、建立线程,Runnable中run方法执行费时操作,完成时通过Handler发送消息(这种方式在之后的网络图片加载会用到),2、使用安卓自带的AsyncTask异步类,这里先考虑第二种。
先说一下AsyncTask异步类,网上复制的,理解。
Android提供了一套专门用于异步处理的类。即:AynsTask类。使用这个类可以为耗时程序开辟一个新线程进行处理,处理完时返回。
其实,AsynTask类就是对Thread类的一个封装,并且加入了一些新的方法。编程时,两者都可以实现同样的功能。本文后面将对AsynTask和Thread进行比较。
1、AsynTask类结构
AsysTask类主要用到的几个内部回调函数有:
doInBackGround()、onPreExecute()、onPostExecute()、onProgressUpdate(),正是这几个回调函数构成了AsynTask类的使用逻辑结构。
注意:每个AsynTask子类必须至少复写doInBackGround()方法。
2、回调逻辑关系
调用关系如下图:
1>主线程调用AsynTask子类实例的execute()方法后,首先会调用onPreExecute()方法。onPreExecute()在主线程中运行,可以用来写一些开始提示代码。
2>之后启动新线程,调用doInBackground()方法,进行异步数据处理。
3>处理完毕之后异步线程结束,在主线程中调用onPostExecute()方法。onPostExecute()可以进行一些结束提示处理。
补充:在doInBackground()方法异步处理的时候,如果希望通知主线程一些数据(如:处理进度)。这时,可以调用publishProgress()方法。这时,主线程会调用AsynTask子类的onProgressUpdate()方法进行处理。
3、各个函数间数据的传递
通过上面的调用关系,我们就可以大概看出一些数据传递关系。如下:
execute()向doInBackground()传递。
doInBackground()的返回值会传递给onPostExecute()。
publishProgress()向progressUpdate()传递。
要点:为了调用关系明确及安全,AsynTask类在继承时要传入3个泛型。第一个泛型对应execute()向doInBackground()的传递类型。第二个泛型对应doInBackground()的返回类型和传递给onPostExecute()的类型。第三个泛型对应publishProgress()向progressUpdate()传递的类型。
传递的数据都是对应类型的数组,数组都是可变长的哦。可以根据具体情况使用。
看这么多,可能不懂还更乱了,现在继承AsynTask来实现一个简单的异步子类。如下:
就这么简单的应用一下先,也不知道应用方法对不对。在上节的下拉刷新中简单运用一下。
运行效果就不贴了,一样的,注意看输出,限制一下网速试试?
安卓中主要有两种实现方式,1、建立线程,Runnable中run方法执行费时操作,完成时通过Handler发送消息(这种方式在之后的网络图片加载会用到),2、使用安卓自带的AsyncTask异步类,这里先考虑第二种。
先说一下AsyncTask异步类,网上复制的,理解。
Android提供了一套专门用于异步处理的类。即:AynsTask类。使用这个类可以为耗时程序开辟一个新线程进行处理,处理完时返回。
其实,AsynTask类就是对Thread类的一个封装,并且加入了一些新的方法。编程时,两者都可以实现同样的功能。本文后面将对AsynTask和Thread进行比较。
1、AsynTask类结构
AsysTask类主要用到的几个内部回调函数有:
doInBackGround()、onPreExecute()、onPostExecute()、onProgressUpdate(),正是这几个回调函数构成了AsynTask类的使用逻辑结构。
注意:每个AsynTask子类必须至少复写doInBackGround()方法。
2、回调逻辑关系
调用关系如下图:
1>主线程调用AsynTask子类实例的execute()方法后,首先会调用onPreExecute()方法。onPreExecute()在主线程中运行,可以用来写一些开始提示代码。
2>之后启动新线程,调用doInBackground()方法,进行异步数据处理。
3>处理完毕之后异步线程结束,在主线程中调用onPostExecute()方法。onPostExecute()可以进行一些结束提示处理。
补充:在doInBackground()方法异步处理的时候,如果希望通知主线程一些数据(如:处理进度)。这时,可以调用publishProgress()方法。这时,主线程会调用AsynTask子类的onProgressUpdate()方法进行处理。
3、各个函数间数据的传递
通过上面的调用关系,我们就可以大概看出一些数据传递关系。如下:
execute()向doInBackground()传递。
doInBackground()的返回值会传递给onPostExecute()。
publishProgress()向progressUpdate()传递。
要点:为了调用关系明确及安全,AsynTask类在继承时要传入3个泛型。第一个泛型对应execute()向doInBackground()的传递类型。第二个泛型对应doInBackground()的返回类型和传递给onPostExecute()的类型。第三个泛型对应publishProgress()向progressUpdate()传递的类型。
传递的数据都是对应类型的数组,数组都是可变长的哦。可以根据具体情况使用。
看这么多,可能不懂还更乱了,现在继承AsynTask来实现一个简单的异步子类。如下:
public class BackgroundTask extends AsyncTask<Void,Void,Void> { BGTaskListener mListener; @Override protected Void doInBackground(Void... params) {//后台处理任务 // TODO Auto-generated method stub if(mListener!=null){ mListener.taskDoing();//具体操作在taskDoing中 } return null; } @Override protected void onPostExecute(Void result) {//任务完成 // TODO Auto-generated method stub if(mListener!=null){ mListener.taskFinish();//完成通知 } super.onPostExecute(result); } public void setTaskListener(BGTaskListener listener){ mListener=listener; } public interface BGTaskListener{ public void taskDoing(); public void taskFinish(); } }
就这么简单的应用一下先,也不知道应用方法对不对。在上节的下拉刷新中简单运用一下。
private QPullListener qplistener=new QPullListener() { @Override public void onRefresh() { // TODO Auto-generated method stub bgtask=new BackgroundTask(); bgtask.setTaskListener(bgtlistener); bgtask.execute(); } … } private BGTaskListener bgtlistener=new BGTaskListener() { @Override public void taskFinish() { // TODO Auto-generated method stub System.out.println(rcvdata); qlv_list.completeRefresh(QListView.ISSUCCESS);//完成刷新 } @Override public void taskDoing() { // TODO Auto-generated method stub rcvdata=Http.read("http://www.baidu.com");//一个自定义网络类,获取百度首页数据 } };
运行效果就不贴了,一样的,注意看输出,限制一下网速试试?
相信自己——2017/05/18