您的位置:首页 > 移动开发 > Android开发

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

2014-01-07 09:42 405 查看

类概述

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文件

[java] view plaincopy

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);

}

@Override

public 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;

@Override

protected void onPreExecute() {

button1.setEnabled(false);

super.onPreExecute();

}

@Override

protected 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 "下载完毕";

}

@Override

protected void onProgressUpdate(Integer... values) {

//这个函数在doInBackground调用publishProgress时触发

textView1.setText(values[0]+"%");

super.onProgressUpdate(values);

}

@Override

protected void onPostExecute(String result) {

//doInBackground返回时触发,换句话说,就是doInBackground执行完后触发

//这里的result就是上面doInBackground执行后的返回值,所以这里是"执行完毕"

setTitle(result);

button1.setEnabled(true);

super.onPostExecute(result);

}

}

}

XML布局文件

[html] view plaincopy

<?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>

效果图

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



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

内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: