android非同步更新UI
在android開發中不能在非ui執行緒中更新ui,但是,有的時候我們需要在程式碼中執行一些諸如訪問網路、查詢資料庫等耗時操作,為了不阻塞ui執行緒,我們時常會開啟一個新的執行緒(工作執行緒)來執行這些耗時操作,然後我們可能需要將查詢到的資料渲染到ui元件上,那麼這個時候我們就需要考慮非同步更新ui的問題了。
android中有下列幾種非同步更新ui的解決辦法:
Activity.runOnUiThread(Runnable)
View.post(Runnable)
View.postDelayed(Runnable, long)
使用handler(執行緒間通訊)(推薦)
AsyncTask(推薦)
下邊這段程式碼
public void onClick(View v) {
new Thread(new Runnable() {
public void run() {
Bitmap bitmap = loadImageFromNetwork("http://example.com/image.png");
mImageView.setImageBitmap(bitmap);
}
}).start();
}
這段程式碼是一個按鈕點選事件的響應方法,當點選了這個按鈕後開啟了一個子執行緒去網路上載入圖片,然後在這個執行緒中給imageView設定了圖片(更新了ui),這段程式碼在非ui執行緒中更新了ui,執行會引發錯誤。
1. Activity.runOnUiThread:
通常,在Activity,我們可以使用Activity的runOnUiThread方法來更新ui。
public void onClick(View v) { new Thread(new Runnable() { public void run() { Bitmap bitmap = loadImageFromNetwork("http://example.com/image.png"); runOnUiThread(new Runnable() { @Override public void run() { mImageView.setImageBitmap(bitmap); } }); } }).start(); }
2. View.post(Runable)
View類及其子類提供了一個post(Runable)方法允許我們將我們要做的操作放到這個匿名Runable物件的run方法中,在這個方法裡面我們可以直接更新ui。
public void onClick(View v) {
new Thread(new Runnable() {
public void run() {
Bitmap bitmap = loadImageFromNetwork("http://example.com/image.png");
imageView.post(new Runnable() {
@Override
public void run() {
mImageView.setImageBitmap(bitmap);
}
});
}
}).start();
}
3. View.postDelayed(Runnable, long)
和View.post(Runable)方法一樣,只是延遲第二個引數指定的時間後執行,而View.post(Runable)是立即執行。
public void onClick(View v) {
new Thread(new Runnable() {
public void run() {
Bitmap bitmap = loadImageFromNetwork("http://example.com/image.png");
imageView.postDelayed(new Runnable() {
@Override
public void run() {
mImageView.setImageBitmap(bitmap);
}
},2000);
}
}).start();
}
4. 使用Handler(推薦)
前面說道的幾種方法當這種操作過多的時候,我們的程式碼會顯得臃腫,程式碼及業務都難於管理控制,所以,當這類程式碼多的時候我們就應該採取Handler的方式了。
new Thread(new Runnable() {
public void run() {
Bitmap bitmap = loadImageFromNetwork("http://example.com/image.png");
Message message = mHandler.obtainMessage();
message.what = 1;
message.obj = bitmap;
mHandler.sendMessage(message);
}
}).start();
Handler mHandler = new Handler(){
@Override
public void handleMessage(Message msg) {
switch (msg.what){
case 1:
Bitmap bitmap = (Bitmap) msg.obj;
imageView.setImageBitmap(bitmap);
break;
case 2:
// ...
break;
default:
break;
}
}
};
5. AsyncTask(推薦)
android為我們提供了非同步任務AsyncTask,我們可以使用AsyncTask輕鬆地實現非同步載入資料及更新ui。
AsyncTask<string,void,bitmap> asyncTask = new AsyncTask<string, bitmap="">() {
/**
* 即將要執行耗時任務時回撥,這裡可以做一些初始化操作
*/
@Override
protected void onPreExecute() {
super.onPreExecute();
}
/**
* 在後臺執行耗時操作,其返回值將作為onPostExecute方法的引數
* @param params
* @return
*/
@Override
protected Bitmap doInBackground(String... params) {
Bitmap bitmap = loadImageFromNetwork(params[0]);
return bitmap;
}
/**
* 當這個非同步任務執行完成後,也就是doInBackground()方法完成後,
* 其方法的返回結果就是這裡的引數
* @param bitmap
*/
@Override
protected void onPostExecute(Bitmap bitmap) {
imageView.setImageBitmap(bitmap);
}
};
asyncTask.execute("http://example.com/image.png");
需要知道的是doInBackground方法工作在工作執行緒中,所以,我們在這個方法裡面執行耗時操作。同時,由於其返回結果會傳遞到onPostExecute方法中,而onPostExecute方法工作在UI執行緒,這樣我們就在這個方法裡面更新ui,達到了非同步更新ui的目的。