Android AsyncTask

AsyncTask 简介

AsyncTask 实际上是一个帮助类,可以让我们很简单的从子线程切换到主线程,去更新UI界面,而我们却又可以在 doInBackground() 方法中异步执行耗时任务。这样的话,我们就不需要频繁的手动切换线程去更新UI了。但是需要注意的是,AsyncTask 类设计的时候仅仅是针对一些从执行到结束至多几秒的任务。为什么会这样呢?因为 AsyncTask并不是为每一 AsyncTask实例单独创建一个线程。相反的,它使用 Executor在单一的后台进程上运行所有 AsyncTask的后台任务。也就是说,每个 AsyncTask任务需要排队执行,所以如果一个长时间运行的 AsyncTask 会阻塞其他AsyncTask 任务。如果想要长时间的任务,推荐使用 java.util.concurrent 包中提供的一些 api,如Executor, ThreadPoolExecutor and FutureTask.

AsyncTask 的三个参数

如果我们想要自定义一个AsyncTask,那我们必须这样定义

private class MyTask extends AsyncTask { ... }

可见,AysncTask 中包含了三个参数,下面来详细介绍

1.Params : 这个参数是规定doInBackground(Params)方法中Params数据类型 的,当然,如果不需要传递任何数据,可以设置为Void,如果需要传递数据,那么在执行时需要在execute()方法中输入参数。

示例 :

AsyncTask task = new AsyncTask() {
        @Override
        protected Void doInBackground(String... params) {
            for (String parameter : params){
                Log.d("MainActivity", parameter);
            }
            return null;
        }
    };
    
task.execute("Hello","Android");

2.Progress : 这个参数规定onProgressUpdate(Progress)方法中的参数类型的,当然如果不需要,可以设置为Void

示例 :

 AsyncTask task = new AsyncTask() {
        @Override
        protected Void doInBackground(Integer... integers) {
            for (Integer progress : integers){
                publishProgress(progress);
            }
            return null;
        }

        @Override
        protected void onProgressUpdate(Integer... values) {
            int progress = values[0];
            progressBar.setProgress(progress);
        }
    };
    
    task.execute(0,25,50,75,100);

3.Result : 这个参数规定 doInBackground()方法的数据返回类型,就不再多说。

AsyncTask 中主要的函数

  1. onPreExecute()

该方法是 AysncTask运行的第一个方法, 可以用于初始化UI,比如说 显示一个进度条等。

  1. doInBackground (Params... params)

这个方法所有代码都会在一个子线程中执行,所有的耗时任务都在这处理,处理完之后通过 return 将结果返回。在这个方法中如果需要进行更新 UI,可以通过 publishProgress(Progress... values)方法,这个方法会调用onProgressUpdate(Progress... values)去进行更新。

  1. onProgressUpdate (Progress... values)

这个方法在publishProgress()方法调用时调用,用于 更新UI界面。

  1. publishProgress (Progress... values)

注意这个方法被设计为终极方法,即final修饰的方法,所以不可在子类中被复写,我们只需要调用该函数触发onProgressUpdate() 函数,完成一些 UI界面上的操作。

  1. onPostExecute(Result result)

doInBackground执行return语句后,就会转到该方法上来,主要用于处理一些善后工作,比如关闭进度条。

AsyncTask 的工作模式

通过AysncTask.cancel(boolean)方法,可以撤销正在运行的 AysncTask,不过我们可以选择温和 和 粗暴两种方式。

  1. AysncTask.cancel(false)

这个方法调用之后,会将isCancelled()状态设置为 true,所以只要我们在doInBackground()方法中频繁的检查这个状态,就可以尽可能快的结束运行。

  1. AysncTask.cancel(true)

该方法会直接终止doInBackground()方法中的所有线程。

使用 AsyncTask 的注意事项

  1. AysncTask只能在 UI 线程中加载并且实例化。
  2. AsyncTask.execute() 必须在UI线程中调用。
  3. 不要手动调用onPreExecute(), onPostExecute(Result), doInBackground(Params...), onProgressUpdate(Progress...)等函数。
  4. 一个任务只能被执行一次,如果两次调用execute(),系统会抛出一个Exception
  5. AysncTask中还有名为 onCancelled()的函数,这个函数的触发条件为AsyncTask.cancel()被调用 且doInBackground() 执行完毕。这也就意味着,当AsyncTask.cancel()调用后,执行完doInBackground() 就会执行onCancelled(),而不是onPostExecute()

使用AsyncTask

public class DownloadTask extends AsyncTask{

private Context mContext;
private ProgressDialog dialog;

public DownloadTask(Context context){
    mContext = context;
}
@Override
protected void onPreExecute() {
    dialog = new ProgressDialog(mContext);
    dialog.setTitle("Downloading");
    dialog.setMax(100);
    dialog.show();
}

@Override
protected Boolean doInBackground(Void... voids) {
    try {
        int percent = 0;
        while(percent <= 100){
            publishProgress(percent);
            percent += 25;
            Thread.sleep(1000);
        }
    }catch (Exception e){
        return false;
    }
    return true;
}

@Override
protected void onProgressUpdate(Integer... values) {
    dialog.setMessage(values[0]+"%");
}

@Override
protected void onPostExecute(Boolean aBoolean) {
    dialog.dismiss();
    if (aBoolean){
        Toast.makeText(mContext, "Download success", Toast.LENGTH_SHORT).show();
    } else {
        Toast.makeText(mContext, "Download failed", Toast.LENGTH_SHORT).show();
    }
  }
}


public class MainActivity extends AppCompatActivity {

private DownloadTask mDownloadTask;
private Button mButton;
@Override
protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main);
    mDownloadTask = new DownloadTask(this);
    mButton = (Button) findViewById(R.id.start);
    mButton.setOnClickListener(new View.OnClickListener() {
        @Override
        public void onClick(View view) {
            mDownloadTask.execute();
        }
    });

}

你可能感兴趣的:(Android AsyncTask)