现在的位置: 首页 > 综合 > 正文

AsyncTask多进程与多线程

2018年02月17日 ⁄ 综合 ⁄ 共 2918字 ⁄ 字号 评论关闭

一、简介

Android系统为了避免与UI线程阻塞,于是便有了Handler和AsyncTask类,由于UI的更新只能在主线程种完成,因此异步处理是不可避免的。

进程:当某个组件第一次运行的时候,Android启动了一个进程。默认的,所有的组件和程序运行在这个进程和线程中。

    线程是程序运行的基本执行单元。当操作系统(不包括单线程的操作系统,如微软早期的DOS)在执行一个程序时,会在系统中建立一个进程,而在这个进程中,必须至少建立一个线程(这个线程被称为主线程)来作为这个程序运行的入口点。因此,在操作系统中运行的任何程序都至少有一个主线程。 

     AsyncTask是个抽象类,是一个工具类,在后台异步的处理任务,不会阻塞UI线程。本质上它底层是个线程池(而Handler底层是个消息队列)。

二、常用方法

AsyncTask构造方法:AsyncTask<Params, Progress, Result>。

    Params 启动任务执行的输入参数,比如HTTP请求的URL

    Progress 后台任务执行的百分比。

    Result 后台执行任务最终返回的结果,比如String。

   必须重写的方法:

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只能被执行一次,否则多次调用时将会出现异常;


下面举个最简单的例子来演示下AsyncTask的用法:

import android.app.Activity;
import android.os.AsyncTask;
import android.os.Bundle;
import android.view.View;
import android.widget.ProgressBar;
import android.widget.TextView;
import android.widget.Toast;

public class AsyncTaskActivity extends Activity
{
	private ProgressBar mProgressBar;
	private TextView mTextView;
	private ProgressAsyncTask mProgressAsyncTask;

	@Override
	protected void onCreate(Bundle savedInstanceState)
	{
		super.onCreate(savedInstanceState);
		setContentView(R.layout.activity_async_task);
		mProgressBar = (ProgressBar) findViewById(R.id.progressBar1);
		mTextView = (TextView) findViewById(R.id.textView1);
	}

	private class ProgressAsyncTask extends AsyncTask<String, Integer, Integer>
	{
        private int mCount;
        /*
         * 这里的String参数对应第一个参数
         * 返回值对应第三个参数
         * 调用publishProgress()触发onProgressUpdate()更新进度
         * */
		@Override
		protected Integer doInBackground(String... params)
		{
			mProgressBar.setMax(params.length);
			mCount = params.length;//5
			
			for (int i = 0; i < mCount; i++)
			{
				publishProgress(i + 1);
				if (isCancelled())
					break;
				try
				{
					Thread.sleep(1000);
				}
				catch (Exception e)
				{

				}
			}

			return params.length;
		}
		/*
		 * 这里的Integer对应第三个参数,也就是doInBackGround()的返回值
		 * */
		@Override
		protected void onPostExecute(Integer result)
		{
			super.onPostExecute(result);
			Toast.makeText(AsyncTaskActivity.this, "任务完成,共处理" + result + "个值.", Toast.LENGTH_LONG)
					.show();

		}
		/*
		 * 这里的Integer对应第二个参数
		 * 在doInBackground()方法中触发此方法
		 * */
		@Override
		protected void onProgressUpdate(Integer... values)
		{
			super.onProgressUpdate(values);
			mProgressBar.setProgress(values[0]);
			mTextView.setText(100 * values[0]/mCount + "%");
		}

		@Override
		protected void onCancelled()
		{
			Toast.makeText(AsyncTaskActivity.this, "任务已取消", Toast.LENGTH_LONG)
					.show();

			super.onCancelled();
		}

	}

	public void onClick_Start(View view)
	{
		mProgressAsyncTask = new ProgressAsyncTask();

		mProgressAsyncTask.execute("str1", "str2", "str3", "str4", "str5");
	}

	public void onClick_Cancel(View view)
	{
		mProgressAsyncTask.cancel(true);
	}
}

【上篇】
【下篇】

抱歉!评论已关闭.