【Android】Handler应用(四):AsyncTask的用法与实例

来源:互联网 发布:京东双十一实时数据 编辑:程序博客网 时间:2024/06/08 14:20

类概述

AsyncTask能够适当地、简单地用于 UI线程。这个类不需要操作线程(Thread)就可以完成后台操作将结果返回UI。

异步任务的定义是一个在后台线程上运行,其结果是在 UI线程上发布的计算。 

异步任务被定义成三种泛型类型: Params,Progress和 Result;

和四个步骤: begin , doInBackground,processProgress 和end。

AsyncTask的泛型类型

这三个类型被用于一个异步任务,如下: 

1. Params,启动任务执行的输入参数 

2. Progress,后台任务执行的百分比 

3. Result,后台计算的结果类型 

在一个异步任务里,不是所有的类型总被用。

假如一个类型不被使用,可以简单地使用 Void类型


AsyncTask的4个步骤

当一个异步任务被执行,任务经过四各步骤: 

1. onPreExecute(),在UI线程上调用任务后立即执行。

这步通常被用于设置任务,例如在用户界面显示一个进度条。 

2. doInBackground(Params...),后台线程执行onPreExecute()完后立即调用。

这步被用于执行较长时间的后台计算。异步任务的参数也被传到这步。

计算的结果必须在这步返回,将传回到上一步。

在执行过程中可以调用publishProgress(Progress...)来更新任务的进度。

3. onProgressUpdate(Progress...),一次呼叫 publishProgress(Progress...)后调用 UI线程。

执行时间是不确定的。这个方法用于当后台计算还在进行时在用户界面显示进度。

例如:这个方法可以被用于一个进度条动画或在文本域显示记录。 

4. onPostExecute(Result), 当后台计算结束时,调用 UI线程。

后台计算结果作为一个参数传递到这步。 

AsyncTask的线程规则

有一些线程规则必须去遵守,这个类才会正确的工作:

1.  任务实例必须创建在 UI线程 

2.  execute(Params...)必须在 UI线程上调用 

3.  不要手动调用onPreExecute(), onPostExecute(Result), doInBackground(Params...), onProgressUpdate(Progress...)

4.  这个任务只执行一次(如果执行第二次将会抛出异常)


AsyncTask使用的注意点

AsyncTask一定要在Ui所在的主线程中创建;

不要手工调用它的四个函数;

只能execute一次;


AsyncTask的应用实例

Activity文件

package com.app.myhandler;import android.app.Activity;import android.os.AsyncTask;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 AsyncTaskDemo extends Activity implements OnClickListener {private TextView textView1;private Button button1;private ProgressBar progressBar1;    @Override    public void onCreate(Bundle savedInstanceState) {        super.onCreate(savedInstanceState);        setContentView(R.layout.activity_asynctask);                progressBar1 = (ProgressBar)findViewById(R.id.progressBar1);        button1 = (Button)findViewById(R.id.button1);        textView1 = (TextView)findViewById(R.id.textView1);                button1.setOnClickListener(this);    } @Overridepublic void onClick(View v) {if(button1.getText().equals("开始")) {setTitle("正在下载...");  new MyTask().execute(1);}}boolean flag = true;@SuppressWarnings("unused")private class MyTask extends AsyncTask<Integer, Integer, String> {int i = 0;@Overrideprotected void onPreExecute() {button1.setEnabled(false);super.onPreExecute();}@Overrideprotected String doInBackground(Integer... params) {//第二个执行方法,onPreExecute()执行完后执行  System.out.println("flag="+flag);while(flag) {i++;if(i<=100) {                progressBar1.setProgress(i);                  publishProgress(i);                  try {                      Thread.sleep(params[0]);                  } catch (InterruptedException e) {                      e.printStackTrace();                  } } else {break;}}            return "下载完毕"; }@Overrideprotected void onProgressUpdate(Integer... values) {//这个函数在doInBackground调用publishProgress时触发            textView1.setText(values[0]+"%");super.onProgressUpdate(values);}@Overrideprotected void onPostExecute(String result) {//doInBackground返回时触发,换句话说,就是doInBackground执行完后触发              //这里的result就是上面doInBackground执行后的返回值,所以这里是"执行完毕"              setTitle(result);              button1.setEnabled(true);            super.onPostExecute(result);  }}}

XML布局文件

<?xml version="1.0" encoding="utf-8"?><LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"    android:layout_width="match_parent"    android:layout_height="match_parent"    android:orientation="vertical"    android:padding="10dp" >    <Button        android:id="@+id/button1"        android:layout_width="wrap_content"        android:layout_height="wrap_content"        android:text="开始"        android:layout_gravity="center_horizontal" />    <ProgressBar        android:id="@+id/progressBar1"        style="?android:attr/progressBarStyleHorizontal"        android:layout_width="match_parent"        android:layout_height="wrap_content" />    <TextView        android:id="@+id/textView1"        android:layout_width="wrap_content"        android:layout_height="wrap_content"        android:text="TextView" /></LinearLayout>

效果图

点击开始,进度条即开始运行(同时按钮被禁用):

进度条加载完成后,按钮解禁,以供再次点击。


原创粉丝点击