在Android中实现异步任务机制有两种方法,Handler和AsyncTask。
Handler模式需要为每一个任务创建一个新的线程,任务完成后通过Handler实例向UI线程发送消息,完成界面的更新,这种方式对于整个过程的控制比较精细,但也是有缺点的,例如代码相对臃肿,在多个任务同时执行的时候,不易对线程进行精确的控制。
为了简化操作,Android1.5提供了工具类android.os.AsynvTask,它使创建异步任务变的更加简单,不在需要编写任务线程和Handler实例即可完成相同的任务。
先来看看AsyncTask的定义:
1、public abstract class AsyncTask<Params,Progress,Result>{}
三种泛型类型分别代表着“启动任务执行的输入参数”、
“后台任务执行的进度”、“后台计算结果的类型”。在特定的场合下,并不是所有类型都被使用,如果没有被使用,可以用java.lang.Void类型来替代。
一个异步任务的执行一般包括以下几个步骤:
1. execute(Params... params),执行一个异步任务,需要我们在代码中调用此方法,触发异步任务的执行。
2. onPreExecute(),在execute(Params... params)被调用后立即被执行,一般用来在执行后台任务前对UI做一些标记。
3. doInBackground(Params... params),在onPreExecute()完成后立即执行,用于执行较为费时的操作,此方法将接收输入参数和返回计算结果。在执行过程中可以调用publishProgress(Progress... values)来更新进度信息。
4. onProgressUpdate(Progress... values),在调用
publishProgress(Progress... values)时,此方法被执行,直接将进度信息更新到UI组件上。
5. onPostExecute(Result result),当后台操作结束的时候,此方法将会被调用,计算结果将作为参数传递到此方法中,直接将结果显示到UI组件上。
在使用的时候,格外的注意一下的几点:
1. 异步任务的实例必须在UI线程中创建。
2. execut(Params... params)方法必须在UI线程中调用。
3. 不要手动的调用onPreExecute(),doInBackground(Params... params),onProgressUpdate(Progress...values),onPostExecute(Result result)这几个方法。
4. 不能再doInBackground(Params ... params)中更新UI组件的信息。
5. 一个任务实例只能执行一次,如果执行第二次将会抛出异常。一个简单的小例子如下:
当然了哈,我并不是用的activity来执行的,而是运用了fragment,通过activity来管理fragment(这个思想是如果你以后用这着,那么你刚开始就要使用它)
package com.example.download; import android.app.AlertDialog; import android.os.AsyncTask; import android.os.Bundle; import android.support.v4.app.Fragment; import android.view.LayoutInflater; import android.view.View; import android.view.ViewGroup; import android.widget.Button; import android.widget.ProgressBar; import android.widget.TextView; public class DownloadFragment extends Fragment { private ProgressBar progressBar; private Button downloadButton; private TextView tv; @Override public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) { View v = inflater.inflate(R.layout.fragment_download, container, false); tv = (TextView) v.findViewById(R.id.tv); downloadButton = (Button) v.findViewById(R.id.download); downloadButton.setOnClickListener(new View.OnClickListener() { @Override public void onClick(View v) { MyTask myTask = new MyTask(); myTask.execute(100); } }); progressBar = (ProgressBar) v.findViewById(R.id.progressBar); return v; } private class MyTask extends AsyncTask<Integer, Integer, String> { @Override protected void onPreExecute() { super.onPreExecute(); } @Override protected String doInBackground(Integer... params) { for (int i = 0; i < 100; i++) { progressBar.setProgress(i); publishProgress(i); try { Thread.sleep(params[0]); } catch (InterruptedException e) { e.printStackTrace(); } } return "执行完毕"; } @Override protected void onProgressUpdate(Integer... values) { super.onProgressUpdate(values); tv.setText(values[0] + "%"); } @Override protected void onPostExecute(String s) { super.onPostExecute(s); tv.setText(s); new AlertDialog.Builder(getActivity()).setTitle("加载完成") .setIcon(R.mipmap.ic_launcher) .setMessage("好了哈").show(); } } }
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:orientation="vertical" android:layout_width="fill_parent" android:layout_height="fill_parent" > <Button android:id="@+id/download" android:layout_width="fill_parent" android:layout_height="wrap_content" android:textAllCaps="false" android:text="Download"/> <TextView android:id="@+id/tv" android:layout_width="fill_parent" android:layout_height="wrap_content" android:text="当前进度显示"/> <ProgressBar android:id="@+id/progressBar" android:layout_width="fill_parent" android:layout_height="40dp" style="?android:attr/progressBarStyleHorizontal"/> </LinearLayout>