AsyncTask 实际上是一个帮助类,可以让我们很简单的从子线程切换到主线程,去更新UI 界面,而我们却又可以在 doInBackground() 方法中异步执行耗时任务。这样的话,我们就不需要频繁的手动切换线程去更新UI了。但是需要注意的是,AsyncTask 类设计的时候仅仅是针对一些从执行到结束至多几秒的任务。为什么会这样呢?因为 AsyncTask 并不是为每一 AsyncTask 实例单独创建一个线程。相反的,它使用 Executor 在单一的后台进程上运行所有 AsyncTask 的后台任务。也就是说,每个 AsyncTask 任务需要排队执行,所以如果一个长时间运行的 AsyncTask 会阻塞其他 AsyncTask 任务。如果想要长时间的任务,推荐使用 java.util.concurrent 包中提供的一些 api,如 Executor, ThreadPoolExecutor and FutureTask.
如果我们想要自定义一个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() 方法的数据返回类型,就不再多说。
该方法是 AysncTask 运行的第一个方法, 可以用于初始化UI,比如说 显示一个进度条等。
这个方法所有代码都会在一个子线程中执行,所有的耗时任务都在这处理,处理完之后通过 return 将结果返回。在这个方法中如果需要进行更新 UI,可以通过 publishProgress(Progress… values)方法,这个方法会调用onProgressUpdate(Progress… values)去进行更新。
这个方法在 publishProgress() 方法调用时调用,用于 更新 UI 界面。
注意这个方法被设计为终极方法,即 final 修饰的方法,所以不可在子类中被复写,我们只需要调用该函数触发 onProgressUpdate() 函数,完成一些 UI 界面上的操作。
doInBackground 执行return语句后,就会转到该方法上来,主要用于处理一些善后工作,比如关闭进度条。
通过AysncTask.cancel(boolean) 方法,可以撤销正在运行的 AysncTask,不过我们可以选择温和 和 粗暴两种方式。
这个方法调用之后,会将 isCancelled() 状态设置为 true,所以只要我们在doInBackground() 方法中频繁的检查这个状态,就可以尽可能快的结束运行。
该方法会直接终止 doInBackground() 方法中的所有线程。
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();
}
});
}