概述
Android 中只有主线程可以操作 ui,以保证 ui 的稳定性,而 Android 是多线程的操作系统,像读取文件、网络操作等耗时较长的操作应该放到子线程中执行,以免阻塞主线程。
利用 AsyncTask 可方便的实现异步操作,既可以在子线程中执行异步操作,又可以更新 ui。
AsyncTask<Params, Progress, Result>
是一个抽象类,用于被继承以实现自定义的异步任务。
继承 AsyncTask 需要指定三个参数类型:
- Params:启动任务时输入参数的类型。
- Progress:后台任务执行中返回进度值的类型。
- Result:后台执行任务完成后返回结果的类型。
AsyncTask 通常需要重写的方法有四个:
- onPreExecute:执行后台耗时操作前被调用,通常用于完成一些初始化操作。运行在主线程。
- doInBackground:必须重写。异步执行后台线程将要完成的任务。
- onProgressUpdate:在doInBackground方法中调用publishProgress方法时调用,用于更新进度。运行在主线程。
- onPostExecute:当doInBackground完成后,系统会将doInBackground的返回值传入该方法,以进行后续操作。运行在主线程。
例子:加载网络图片
activity_load_image.xml
<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http:///apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:padding="16dp">
<ImageView
android:layout_width="match_parent"
android:layout_height="match_parent"
android:id="@+id/image"/>
<ProgressBar
android:visibility="gone"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_centerInParent="true"
android:id="@+id/progressbar"/>
</RelativeLayout>
LoadImageActivity.java
public class LoadImageActivity extends Activity {
/**
* 网络图片的地址
*/
private static final String URL = "http://imgsrc.baidu.com/forum/w=580/sign=57ae714ad2160924dc25a213e407359b/8a82b9014a90f60392de8a5b3b12b31bb051ed44.jpg";
private ImageView imageView;
private ProgressBar progressBar;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_load_image);
imageView = (ImageView) findViewById(.image);
progressBar = (ProgressBar) findViewById(.progressbar);
//启动异步任务
new LoadImageAsyncTask().execute(URL);
}
class LoadImageAsyncTask extends AsyncTask<String, Void, Bitmap> {
/**
* 异步任务的准备工作(运行在ui线程)
*/
@Override
protected void onPreExecute() {
super.onPreExecute();
//显示进度条
progressBar.setVisibility(View.VISIBLE);
}
/**
* 异步任务(运行在异步线程)
* @param strings
* @return
*/
@Override
protected Bitmap doInBackground(String... strings) {
String url = strings[0];
Bitmap bitmap = null;
if (url != null && url.isEmpty() == false) {
//下载网络图片,转换为Bitmap
URLConnection urlConnection = null;
InputStream is = null;
try {
urlConnection = new URL(url).openConnection();
is = urlConnection.getInputStream();
BufferedInputStream bis = new BufferedInputStream(is);
bitmap = BitmapFactory.decodeStream(bis);
is.close();
bis.close();
} catch (IOException e) {
e.printStackTrace();
}
}
return bitmap;
}
/**
* 异步任务的结果处理(运行在ui线程)
* @param bitmap
*/
@Override
protected void onPostExecute(Bitmap bitmap) {
super.onPostExecute(bitmap);
//隐藏进度条
progressBar.setVisibility(View.GONE);
//显示图片
if (bitmap != null) {
imageView.setImageBitmap(bitmap);
}
}
}
}
AndroidManifest.xml
<uses-permission android:name="android.permission.INTERNET"/>
例子:更新进度条 & 取消异步任务
activity_load_progressbar.xml
<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http:///apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:padding="16dp">
<ProgressBar
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_centerInParent="true"
android:id="@+id/progressbar"
style="?android:attr/progressBarStyleHorizontal"/>
</RelativeLayout>
LoadProgressbarActivity.java
public class LoadProgressbarActivity extends Activity {
private ProgressBar progressBar;
private LoadProgressbarAsyncTask asyncTask;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_load_progressbar);
progressBar = (ProgressBar) findViewById(.progressbar);
asyncTask = new LoadProgressbarAsyncTask();
asyncTask.execute();
}
@Override
protected void onPause() {
if (asyncTask != null && asyncTask.getStatus() == AsyncTask.Status.RUNNING) {
//标记为取消状态,否则在退出activity再进入activity时,要等待前一个任务执行完成才会开始执行
asyncTask.cancel(true);
}
super.onPause();
}
class LoadProgressbarAsyncTask extends AsyncTask<Void, Integer, Void> {
@Override
protected Void doInBackground(Void... voids) {
for (int i=0; i<100; i++) {
//判断若已取消,则停止
if (isCancelled()) {
break;
}
try {
Thread.sleep(300);
} catch (InterruptedException e) {
e.printStackTrace();
}
//更新进度
publishProgress(i);
}
return null;
}
@Override
protected void onProgressUpdate(Integer... values) {
super.onProgressUpdate(values);
if (isCancelled()) {
return;
}
//更新进度条
progressBar.setProgress(values[0]);
}
}
}
AsyncTask 注意事项
- 必须在 ui 线程中创建 AsyncTask 的实例。
- 必须在 ui 线程中调用 execute 方法。
- 重写的四个方法是系统自动调用的,不应手动调用。
- 每个 AsyncTask 只能被执行一次,多次调用会引发异常。
- doInBackground 中不能直接更新 ui。
参考:http://www.imooc.com/learn/377