Android AsyncTask usage

1.概念

在之前的一篇博客:Android Intent Service (http://blog.csdn.net/afandaafandaafanda/article/details/47128741)中讲到,使用Intent Service 可以执行异步任务,避免由于service与activity都是在主线程,造成service超过5秒无法返回结果而引发的ANR错误。其实,Android还提供了一种机制,那就是AsyncTask组件,这个组件能够很多地完成异步任务,非常类似于.NET中的Task。

AsyncTask,是android提供的轻量级的异步类,可以直接继承AsyncTask,在类中实现异步操作,并提供接口反馈当前异步执行的程度(可以通过接口实现UI进度更新),最后反馈执行的结果给UI主线程.AsyncTask使用起来非常简单,如果使用过.NET中的BackGroundWorker组件的话,就能够非常清楚地了解AsyncTask的机制,使用起来相信非常快。但AsyncTask也并非是有利无害。在使用多个异步操作和并需要进行Ui变更时,就变得复杂起来.

AsyncTask直接继承于Object类,位置为android.os.AsyncTask。要使用AsyncTask工作我们要提供三个泛型参数,并重载几个方法(至少重载一个)。
AsyncTask定义了三种泛型类型 Params,Progress和Result。
Params 启动任务执行的输入参数,比如HTTP请求的URL。
Progress 后台任务执行的百分比。
Result 后台执行任务最终返回的结果,比如String。
使用过AsyncTask 的同学都知道一个异步加载数据最少要重写以下这两个方法:

doInBackground(Params…) 后台执行,比较耗时的操作都可以放在这里。注意这里不能直接操作UI。此方法在后台线程执行,完成任务的主要工作,通常需要较长的时间。在执行过程中可以调用publicProgress(Progress…)来更新任务的进度。
onPostExecute(Result)  相当于Handler 处理UI的方式,在这里面可以使用在doInBackground 得到的结果处理操作UI。 此方法在主线程执行,任务执行的结果作为此方法的参数返回
有必要的话你还得重写以下这三个方法,但不是必须的:

onProgressUpdate(Progress…)   可以使用进度条增加用户体验度。 此方法在主线程执行,用于显示任务执行的进度。
onPreExecute()        这里是最终用户调用Excute时的接口,当任务执行之前开始调用此方法,可以在这里显示进度对话框。
onCancelled()             用户调用取消时,要做的操作
使用AsyncTask类,以下是几条必须遵守的准则:

Task的实例必须在UI thread中创建;
execute方法必须在UI thread中调用;
不要手动的调用onPreExecute(), onPostExecute(Result),doInBackground(Params...), onProgressUpdate(Progress...)这几个方法;
该task只能被执行一次,否则多次调用时将会出现异常;

2.实例

1)MainActivity

import java.util.Date;
import android.app.Activity;
import android.app.ProgressDialog;
import android.os.AsyncTask;
import android.os.Bundle;
import android.util.Log;
import android.view.View;
import android.view.View.OnClickListener;
import android.widget.Button;
import android.widget.TextView;

public class MainActivity extends Activity{
	
	private TextView textView1;
	private Button btn;
	
    @Override
    protected void onCreate(Bundle savedInstanceState){
    	
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        
        textView1 = (TextView)findViewById(R.id.textView1);
        btn = (Button)this.findViewById(R.id.btnAsync);
        btn.setOnClickListener(new OnClickListener(){

			@SuppressWarnings("deprecation")
			@Override
			public void onClick(View arg0) {
				// TODO Auto-generated method stub
				
				Date datestr = new Date();
				StringBuilder stringBuilder = new StringBuilder();
				stringBuilder.append("str at ");
				stringBuilder.append(datestr.getHours());
				stringBuilder.append(":");
				stringBuilder.append(datestr.getMinutes());
				stringBuilder.append(":");
				stringBuilder.append(datestr.getSeconds());
				String str1 = stringBuilder.toString();
				Log.i("AsyncTask", str1);
				
				new DemoThread().execute();
			
				Date dateend = new Date();
				StringBuilder stringBuilder2 = new StringBuilder();
				stringBuilder2.append("str at ");
				stringBuilder2.append(dateend.getHours());
				stringBuilder2.append(":");
				stringBuilder2.append(dateend.getMinutes());
				stringBuilder2.append(":");
				stringBuilder2.append(dateend.getSeconds());
				String str2 = stringBuilder2.toString();
				Log.i("AsyncTask", str2);
			}
        	
        });
    }
    
    final class DemoThread extends AsyncTask<String, String, String>{
    	
    	ProgressDialog pDialog = null;
    	
    	@Override
    	protected String doInBackground(String... arg0) {
    		try {
				Thread.sleep(10*1000);
			} catch (InterruptedException e) {
				// TODO Auto-generated catch block
				e.printStackTrace();
			}
        	return "OK";
    	}
    	
    	@Override
    	protected void onPreExecute() {
        	pDialog = new ProgressDialog(MainActivity.this);
        	pDialog.setProgressStyle(ProgressDialog.STYLE_SPINNER);
        	pDialog.setMessage("please wait。。。");
        	pDialog.setIndeterminate(false);
        	pDialog.setCancelable(true);
        	pDialog.show();
    	}
    	
    	@Override
    	protected void onPostExecute(String result){
    		if (pDialog != null && pDialog.isShowing()){
    			pDialog.hide();
    			pDialog.dismiss();
    		}
    		Log.i("AsyncTask", result);
    		textView1.setText(result);
    	}
    }
}
2)布局文件

<LinearLayout
    xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    android:id="@+id/container"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    tools:context="com.example.asynctask_demo.MainActivity"
    tools:ignore="MergeRootFrame" >
	   <Button
     	android:layout_width="wrap_content" 
	    android:layout_height="wrap_content" 
	    android:text="start Async"
	    android:id="@+id/btnAsync"/>
	   <TextView
            android:id="@+id/textView1"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:text="result will show here。" />
 
</LinearLayout>
3)效果

开始前


异步任务执行中

Android AsyncTask usage_第1张图片

任务执行后

Android AsyncTask usage_第2张图片

执行过程

Android AsyncTask usage_第3张图片

分析:在OnClick时间中,分别记录了起始时间和结束时间,前后相隔不到一面,而最终返回结果的时间是在10秒之后,因为我们的任务是线程停10秒。从日志来看,确实是异步线程在执行,并且获取到了异步线程执行的结果。

你可能感兴趣的:(线程,android,AsyncTask)