Android笔记二十六.Android异步任务处理(AsyncTask)

转载请表明出处:http://blog.csdn.net/u012637501(嵌入式_小J的天空)
一、引言
    我们知道Android的UI线程主要负责处理用户的按键事件用户触屏事件屏幕绘图事件等,对于其他的操作尽量不要在UI线程中实现,因为这些操作很有可能会阻塞UI线程,比如一些耗时操作,会导致UI界面停止响应,从而降低了用户的体验。所以,为了避免UI线程失去响应的问题,Android建议将耗时操作放在新线程中完成,但新线程也可能需要动态更新UI组件:比如需要从网上获取一个网页,然后在TextView中将其源代码显示出来,此时就应该将连接网络,获取网络数据的操作(耗时)放在新线程中完成。
    但是,问题来了:当获取网络数据之后,由于新线程不允许直接更新UI组件,我们该如何更新UI组件数据呢?为了解决新线程不能更新UI组件问题,Android提供了如下几种解决方案:
(1)使用Handler消息传递机制实现线程之间的通信(如上一篇文章所述);
(2)Activity.runOnUiThread(Runnable);
(3)View.post(Runnable);
(4)View.postDelayed(Runnable long);
(5)异步任务。又由于(2)~(4)方式有可能导致编程有点繁琐,而异步任务则可简化这种操作。
二、AsyncTask简介
    Android的类AsyncTask对线程间通讯进行了包装,提供了简易的编程方式使后台线程和UI线程进行通讯:后台线程执行异步任务,并把操作结果通知UI。不再需要子线程和Handler就可以完成异步操作并且刷新用户界面
1.AsyncTask类
    AsyncTask<>是一个抽象类,通常用于被继承,继承AsyncTask时需要指定如下三种泛型参数:
(1)Params:启动任务执行输入参数的类型;
(2)Progress:后台任务完成的进度值(百分比)的类型;
(3)Result:后台执行任务完成后返回结果的类型,如String、Integer;
2.Async类中主要方法
(1)onPreExecute():该方法将在执行实际的后台操作前UI线程调用。可以在该方法中做一些准备工作,如在界面上显示一个进度条,或者一些控件的实例化,这个方法可以不用实现。
-----(UI线程调用,后台线程未启动,初始化工作)
(2)doInBackground(Params...values):将在onPreExecute 方法执行后马上执行,该方法运行在后台线程中。这里将主要负责执行那些比较耗时的后台处理工作。可以调用 publishProgress()方法来实时更新任务进度。该方法是抽象方法,子类必须实现。
-----(后台线程调用,执行后台任务)
(3)onProgressUpdate(Progress...values):在publishProgress方法被调用后,UI 线程将调用这个方法从而在界面上展示任务的进展情况,例如通过一个进度条进行展示。
-----(UI线程调用,展现后台任务进展)
(4)onPostExecute(Result):在doInBackground 执行完成后,onPostExecute 方法将被UI线程调用,后台的计算结果(doInBackground方法返回值)将通过该方法传递到UI线程,并且在界面上展示给用户。
-----(UI线程调用,显示后台线程运行结果)
(5) onCancelled():在用户取消线程操作的时候调用。在主线程中调用onCancelled()的时候调用。
-----(UI线程调用,取消后台线程) 
三、异步任务处理开发步骤
1.创建AsyncTask的子类,并为三个泛型参数指定类型。如果某个泛型参数不需要指定类型,则将它指定为Void。
2.根据需要实现AsyncTask上述五中方法;
3.调用AsyncTask子类的实例的execute(Params... params)开始执行耗时任务。
另外,使用AsyncTask类需要遵守的以下准则
(1)Task的实例必须在UI线程中创建;
(2)execute(Params...)方法必须在UI线程中调用;
(3)不要手动的调用onPreExecute(), onPostExecute(Result),doInBackground(Params...),onProgressUpdate(Progress...)这几个方法,需要在UI线程中实例化这个task来调用;
(4)该task只能被执行一次,否者多次调用时将会出现异常
四、源码实战
1.实现功能
技术分享
2.源码实现
(1)MainActivity.java
功能:用以获取界面组件,实例化一个AsyncTask子类对象并调用其exeute(Integer....params)方法以指定参数params运行一个任务。
package com.example.android_asynctask;
import android.app.Activity;
import android.os.Bundle;
import android.view.View;
import android.view.View.OnClickListener;
import android.widget.Button;
import android.widget.ProgressBar;
import android.widget.TextView;
public class MainActivity extends Activity {
 private Button downbtn;
 private TextView textView;
 private ProgressBar progressBar;
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.main);
        downbtn = (Button)findViewById(R.id.download);
        textView = (TextView)findViewById(R.id.text);
        progressBar = (ProgressBar)findViewById(R.id.progressBar);
        final DownloadTest download = new DownloadTest(textView,progressBar);	//获取一个DownloadTest对象,并传递组件对象参数
        downbtn.setOnClickListener(new OnClickListener()
        {
   @Override
   public void onClick(View v) {
    download.execute(200);
   }
        });
       
    }
}
(2)DownloadTest.java
功能:继承于AsyncTask的子类,其中onPreExecute()作用是为运行后台线程(子线程)完成初始化工作;doInBackground方法运行后台线程;onProgressUpdate方法更新UI;onPostExecute方法处理后台线程运行结果。
package com.example.android_asynctask;
import android.graphics.Color;
import android.os.AsyncTask;
import android.view.View;
import android.widget.ProgressBar;
import android.widget.TextView;
public class DownloadTest extends AsyncTask<Integer,Integer,String>
{
 private TextView tv;
 private ProgressBar pb;
 //带参数构造方法
 DownloadTest(TextView text,ProgressBar bar)
 {
  this.tv = text;
  this.pb = bar;
 }
 //不带参数构造方法
 DownloadTest()
 {
 }
 /*1.onPreExecute方法
  * 为子线程(后台)运行初始化相关内容
  */
 protected void onPreExecute() {
      tv.setVisibility(View.VISIBLE);	 //设置显示文本组件
      pb.setVisibility(View.VISIBLE);	 //设置显示进度条
      super.onPreExecute();
 }
 /*2.doInBackground方法
  *  运行一个后台线程,该线程实现每arg0[0]毫秒调用一次onProgressUpdate方法
  */
 protected String doInBackground(Integer... arg0) {
  for(int i=0;i<100;i++)
  {
   publishProgress(i);	 //调用onProgressUpdate方法并传递参数i
   try {
    Thread.sleep(arg0[0]);	//累加一次,线程休眠argo[0]毫秒
   } catch (InterruptedException e) {
    e.printStackTrace();
   }
  }
  return "下载完毕";	 //后台子线程运行完毕后,返回的值
 }
 /*3.onProgressUpdate方法
  * 调用publishProgress(i)时调用该方法,并传递参数i给形参values[0]*/

 @Override
 protected void onProgressUpdate(Integer... values) {
  pb.setProgress(values[0]);	 //设置进度条值
  tv.setText("已经下载"+values[0]+"%");	 //文本组件显示提示信息
  super.onProgressUpdate(values);
 }
 /*4.onPostExecute
  * 处理后台线程得到的结果
  * */
 protected void onPostExecute(String result) {
  pb.setVisibility(View.INVISIBLE);	//隐藏进度条
  tv.setVisibility(View.VISIBLE);	//显示UI文本显示框组件
  tv.setText(result);
  tv.setTextSize(20);
  tv.setTextColor(Color.RED);
  super.onPostExecute(result);
 }
}


效果演示:
技术分享
技术分享
源码分析:
    通过源码我们可以知道,主线程通过调用AsyncTask子类的execute()方法,进而调用AsyncTask子类的onPreExecute方法,用以再运行后台线程之前为其初始化相关内容。然后,在新创建的子线程中调用doInBackground()方法实现后台线程功能并通过publishProgress()方法调用onProgressUpdate()方法更新UI内容,最后在主线中执行onPostExecute方法来处理后台线程运行的结果。
技术分享
注意:其中只有doInBackground()方法,以及publishProgress()方法是在子线程中执行的,其他的方法都是在主线程中执行的,所以可以在这些方法中更新界面组件。

郑重声明:本站内容如果来自互联网及其他传播媒体,其版权均属原媒体及文章作者所有。转载目的在于传递更多信息及用于网络分享,并不代表本站赞同其观点和对其真实性负责,也不构成任何其他建议。