Android開發(fā)中避免應(yīng)用無響應(yīng)的方法(Application Not Responding、ANR)
App里發(fā)生的最糟糕的事是彈出應(yīng)用無響應(yīng)”Application Not Responding” (ANR) 對話框.本課講的是如何保持應(yīng)用響應(yīng),避免ANR。
什么觸發(fā)ANR
通常,系統(tǒng)會在應(yīng)用無法對用戶輸入響應(yīng)時顯示ANR。比如,如果一個應(yīng)用在I/O操作上阻塞了(頻繁請求網(wǎng)絡(luò))UI線程,系統(tǒng)無法處理用戶輸入事件?;蛘?,在UI線程中,app花了大量時間在構(gòu)建復雜的類,或在游戲中計算下一個動作。保證這些操作高效是很重要的,但最高效的代碼也需要花費時間。
在任何情況下,都不要在UI線程執(zhí)行耗時任務(wù),取而代之的是創(chuàng)建 一個工作線程,在這個線程里操作。這可以保持UI線程運行,阻止系統(tǒng)因為代碼卡住而結(jié)束應(yīng)用。
在Android里,Activity Manager和Window Manager系統(tǒng)服務(wù)監(jiān)控著應(yīng)用的響應(yīng)能力。Android會在檢測到以下情形中之一時,彈出ANR對話框:
1.未在5秒內(nèi)對用戶輸入事件響應(yīng)
2.BroadcastReceiver未在10秒內(nèi)執(zhí)行完
如何避免ANR
Android應(yīng)用默認運行在單線程里,叫UI線程或主線程。這意味著,你的應(yīng)用所有工作都在UI線程里,如果花費很長時間才能完成,會觸發(fā)ANR,因為此時應(yīng)用無法操控輸入事件或廣播。
因此,UI 線程里的任何方法都應(yīng)該盡可能地做輕量的工作,特別是Activity在生命周期方法,像onCreate(),onResume().潛在的耗時操作,像網(wǎng)絡(luò),數(shù)據(jù)庫,或昂貴的計算(像改變圖片大小)應(yīng)該在工作線程里完成(或者在數(shù)據(jù)庫操作案例里,通過一個異步請求)。
最高效的方法是為耗時操作使用AsyncTask類創(chuàng)建工作線程。繼承AsyncTask實現(xiàn)doInBackground()方法來執(zhí)行工作。要發(fā)送進度給用戶,調(diào)用 publishProgress(),會觸發(fā)onProgressUpdate(),例子:
private class DownloadFilesTask extends AsyncTask<URL, Integer, Long> {
// Do the long-running work in here
protected Long doInBackground(URL... urls) {
int count = urls.length;
long totalSize = 0;
for (int i = 0; i < count; i++) {
totalSize += Downloader.downloadFile(urls[i]);
publishProgress((int) ((i / (float) count) * 100));
// Escape early if cancel() is called
if (isCancelled()) break;
}
return totalSize;
}
// This is called each time you call publishProgress()
protected void onProgressUpdate(Integer... progress) {
setProgressPercent(progress[0]);
}
// This is called when doInBackground() is finished
protected void onPostExecute(Long result) {
showNotification("Downloaded " + result + " bytes");
}
}
執(zhí)行這個工作線程,只需要創(chuàng)建一個實例,調(diào)用 execute():
盡管比AsyncTask更復雜,你可能還是想創(chuàng)建自己的線程或者HandlerThread類,如果這么做,你應(yīng)該調(diào)用Process.setThreadPriority(THREAD_PRIORITY_BACKGROUND) 設(shè)置線程優(yōu)先線為”background”.如果沒有,線程仍然會拖慢應(yīng)用,因為它跟UI線程優(yōu)先級相同。
如果你實現(xiàn)Thread或HandlerThread,確保UI線程沒有因為等待工作線程執(zhí)行完而阻塞。不要調(diào)用Thread.wait()或Thread.sleep(),而是提供一個Handler,供任務(wù)執(zhí)行完后回調(diào)。如此設(shè)計,UI線程會保持響應(yīng),避免出現(xiàn)ANR對話框。
特別強調(diào)BroadcastReceiver的執(zhí)行時間,意味著你要:分散工作到后臺線程里,像保存設(shè)置或者注冊Notification。執(zhí)行密集任務(wù)(intensive tasks),應(yīng)該用IntentService。
提示:你可以用StrictMode幫你找到在UI線程上潛在的耗時操作
相關(guān)文章
Android Handler內(nèi)存泄漏原因及解決方案
這篇文章主要介紹了Android Handler內(nèi)存泄漏原因及解決方案,幫助大家更好的理解和利用Android進行開發(fā),感興趣的朋友可以了解下2021-02-02一文詳解?Compose?Navigation?的實現(xiàn)原理
這篇文章主要介紹了一文詳解?Compose?Navigation的實現(xiàn)原理,文章通告圍繞主題展開詳細的相關(guān)內(nèi)容介紹,具有一定的參考價值,需要的小伙伴可以參考一下2022-08-08Android仿微信Viewpager-Fragment惰性加載(lazy-loading)
這篇文章主要為大家詳細介紹了Android仿微信Viewpager-Fragment惰性加載lazy-loading,具有一定的參考價值,感興趣的小伙伴們可以參考一下2017-08-08