基于Android的移動(dòng)護(hù)理異步任務(wù)處理解決方案
移動(dòng)護(hù)理系統(tǒng)開(kāi)發(fā)采用異步處理的方式,可以縮短執(zhí)行操作的時(shí)間,避免UI線程阻塞。筆者介紹了采用異步處理方式開(kāi)發(fā)移動(dòng)護(hù)理程序的方法,并以移動(dòng)護(hù)理中的病人列表異步任務(wù)處理為例進(jìn)行分析。
任務(wù)分析
移動(dòng)護(hù)理開(kāi)發(fā)初期,對(duì)一些比較耗時(shí)的操作,我們沒(méi)有考慮異步任務(wù)處理的方式,造成程序假死的問(wèn)題,用戶(hù)執(zhí)行某個(gè)操作后十幾秒程序才有反應(yīng),嚴(yán)重影響用戶(hù)體驗(yàn)?,F(xiàn)在程序已支持異步任務(wù)處理,本文將簡(jiǎn)單介紹移動(dòng)護(hù)理中異步任務(wù)處理方式。
移動(dòng)護(hù)理系統(tǒng)通過(guò)訪問(wèn)網(wǎng)絡(luò)獲取數(shù)據(jù),在獲取數(shù)據(jù)過(guò)程中,訪問(wèn)網(wǎng)絡(luò)和解析大量XML數(shù)據(jù)是不可避免的,這個(gè)過(guò)程需要耗費(fèi)較長(zhǎng)的時(shí)間,在我們未采用異步任務(wù)處理之前,執(zhí)行一個(gè)操作需要等待5-10秒甚至更長(zhǎng)的時(shí)間,這樣程序就得等待很久才能正常操作。一般情況下,網(wǎng)絡(luò)連接、XML解析、數(shù)據(jù)庫(kù)訪問(wèn)都會(huì)造成這種長(zhǎng)時(shí)間等待。
原因很簡(jiǎn)單,當(dāng)一個(gè)程序第一次啟動(dòng)時(shí),Android會(huì)同時(shí)啟動(dòng)一個(gè)對(duì)應(yīng)的主線程,主線程主要負(fù)責(zé)處理與UI相關(guān)的事件,如:用戶(hù)的按鍵事件、用戶(hù)觸摸屏幕的事件等,并把相關(guān)的時(shí)間分發(fā)到對(duì)應(yīng)的組件進(jìn)行處理,這個(gè)主線程也叫UI線程。我們之前的方式就是使用UI線程完成所有操作,包括UI事件處理、網(wǎng)絡(luò)連接、解析XML,這樣就會(huì)造成UI阻塞,表現(xiàn)為程序假死的現(xiàn)象。
為了解決程序假死的問(wèn)題,我們必須使用異步任務(wù)處理,以確保在UI線程只處理UI事件,避免UI線程阻塞。
解決方案
Android 1.5已經(jīng)提供了一個(gè)工具類(lèi)——AsyncTask,使用AsyncTask可以是創(chuàng)建需要與用戶(hù)界面交互的長(zhǎng)時(shí)間運(yùn)行的任務(wù)變得更簡(jiǎn)單,所以我們使用它來(lái)處理網(wǎng)絡(luò)訪問(wèn)、XML解析這兩個(gè)耗時(shí)操作來(lái)避免移動(dòng)護(hù)理程序假死的問(wèn)題。
AsyncTask 是一個(gè)抽象類(lèi),他提供了三個(gè)泛型參數(shù),詳細(xì)如下
AsyncTask的三個(gè)泛型參數(shù)
1.Param 任務(wù)執(zhí)行需要的參數(shù)
2.Progress 后臺(tái)任務(wù)執(zhí)行的進(jìn)度單位數(shù)據(jù)
3.Result 后臺(tái)任務(wù)執(zhí)行最終返回的數(shù)據(jù)類(lèi)型
注意:在設(shè)置參數(shù)時(shí)通暢是:String …params,這表示方法可以有0個(gè)或多個(gè)String類(lèi)型的參數(shù);不使用參數(shù)時(shí)可設(shè)置為 Void…
使用AsyncTask類(lèi),必須重載AsyncTask的四個(gè)方法(至少重載一個(gè))。詳細(xì)如下:
AsyncTask的四個(gè)方法
1.onPreExecute() 這個(gè)方法主要用于執(zhí)行一些預(yù)處理操作,它運(yùn)行于UI線程,一般用來(lái)為后臺(tái)任務(wù)做一些準(zhǔn)備工作,如在界面上顯示一個(gè)進(jìn)度條。
2.doInBackground() 這個(gè)方法運(yùn)行在后臺(tái)線程中,主要負(fù)責(zé)執(zhí)行那些很耗時(shí)的操作,如移動(dòng)護(hù)理系統(tǒng)中的網(wǎng)絡(luò)連接、解析XML等操作。這個(gè)方法在onPreExecute()方法后執(zhí)行,該方法是抽象方法,也是AsyncTask的關(guān)鍵,所以該方法必須重載。另外,在這個(gè)方法中可以使用 publishProgress(Progress…)來(lái)改變當(dāng)前的進(jìn)度值。
3.doProgressUpdate(Progress…) 這個(gè)方法運(yùn)行于UI線程。如果在doInBackground(Params…)中使用了publishProgress(Progress…),UI線程就會(huì)調(diào)用這個(gè)方法對(duì)進(jìn)度條控件的進(jìn)度值進(jìn)行控制。
4.onPostExecute(Result) 這個(gè)方法也運(yùn)行于UI線程,在doInBackground(Params…)方法執(zhí)行后調(diào)用,該方法用于處理后臺(tái)任務(wù)執(zhí)行后返回的結(jié)果。
實(shí)例解析
以移動(dòng)護(hù)理中的病人列表異步任務(wù)處理為例:
@Override
protected void onPreExecute() {
super.onPreExecute();
// 獲取Wifi網(wǎng)絡(luò)連接狀態(tài)
isNetWorkConnected = NetWorkActivity
.isNetworkAvailable(PatiListActivity.this);
if (isNetWorkConnected)
// 彈出提示對(duì)話框
mDialog.show();
}
onPreExecute()方法主要用于檢查網(wǎng)絡(luò)連接是否成功,如果成功則彈出進(jìn)度條對(duì)話框。該方法運(yùn)行于UI線程。
@Override
protected Object doInBackground(Object... arg0) {
if (isNetWorkConnected) {
// 獲取病人列表數(shù)據(jù)
patientList = getPatiList();
}
return null;
}
doInBackground()方法主要進(jìn)行網(wǎng)絡(luò)訪問(wèn)和XML解析,以獲取病人列表數(shù)據(jù),它運(yùn)行于后臺(tái)線程。這里沒(méi)有調(diào)用 publishProgress() 方法,因?yàn)槲覀兊倪M(jìn)度條不需要顯示百分比,因此我們也沒(méi)必要重載doProgressUpdate(Progress…)方法。
@Override
protected void onPostExecute(Object result) {
super.onPostExecute(result);
if (isNetWorkConnected) {
try {
// 加載頁(yè)面控件
drawPage();
// 關(guān)閉提示對(duì)話框
mDialog.dismiss();
} catch (Exception e) {
e.printStackTrace();
Toast.makeText(PatiListActivity.this, errorMsg,
Toast.LENGTH_SHORT).show();
}
}
}
OnPostExecute()運(yùn)行于UI線程,通過(guò)doInBackgroud()方法獲取的病人列表來(lái)加載頁(yè)面控件,完成后關(guān)閉對(duì)話框。至此,異步任務(wù)處理整個(gè)過(guò)程結(jié)束。
評(píng)論