Android開發(fā)中避免應(yīng)用無(wú)響應(yīng)的方法(Application Not Responding、ANR)
App里發(fā)生的最糟糕的事是彈出應(yīng)用無(wú)響應(yīng)”Application Not Responding” (ANR) 對(duì)話框.本課講的是如何保持應(yīng)用響應(yīng),避免ANR。
什么觸發(fā)ANR
通常,系統(tǒng)會(huì)在應(yīng)用無(wú)法對(duì)用戶輸入響應(yīng)時(shí)顯示ANR。比如,如果一個(gè)應(yīng)用在I/O操作上阻塞了(頻繁請(qǐng)求網(wǎng)絡(luò))UI線程,系統(tǒng)無(wú)法處理用戶輸入事件?;蛘撸赨I線程中,app花了大量時(shí)間在構(gòu)建復(fù)雜的類,或在游戲中計(jì)算下一個(gè)動(dòng)作。保證這些操作高效是很重要的,但最高效的代碼也需要花費(fèi)時(shí)間。
在任何情況下,都不要在UI線程執(zhí)行耗時(shí)任務(wù),取而代之的是創(chuàng)建 一個(gè)工作線程,在這個(gè)線程里操作。這可以保持UI線程運(yùn)行,阻止系統(tǒng)因?yàn)榇a卡住而結(jié)束應(yīng)用。
在Android里,Activity Manager和Window Manager系統(tǒng)服務(wù)監(jiān)控著應(yīng)用的響應(yīng)能力。Android會(huì)在檢測(cè)到以下情形中之一時(shí),彈出ANR對(duì)話框:
1.未在5秒內(nèi)對(duì)用戶輸入事件響應(yīng)
2.BroadcastReceiver未在10秒內(nèi)執(zhí)行完
如何避免ANR
Android應(yīng)用默認(rèn)運(yùn)行在單線程里,叫UI線程或主線程。這意味著,你的應(yīng)用所有工作都在UI線程里,如果花費(fèi)很長(zhǎng)時(shí)間才能完成,會(huì)觸發(fā)ANR,因?yàn)榇藭r(shí)應(yīng)用無(wú)法操控輸入事件或廣播。
因此,UI 線程里的任何方法都應(yīng)該盡可能地做輕量的工作,特別是Activity在生命周期方法,像onCreate(),onResume().潛在的耗時(shí)操作,像網(wǎng)絡(luò),數(shù)據(jù)庫(kù),或昂貴的計(jì)算(像改變圖片大小)應(yīng)該在工作線程里完成(或者在數(shù)據(jù)庫(kù)操作案例里,通過一個(gè)異步請(qǐng)求)。
最高效的方法是為耗時(shí)操作使用AsyncTask類創(chuàng)建工作線程。繼承AsyncTask實(shí)現(xiàn)doInBackground()方法來(lái)執(zhí)行工作。要發(fā)送進(jìn)度給用戶,調(diào)用 publishProgress(),會(huì)觸發(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í)行這個(gè)工作線程,只需要?jiǎng)?chuàng)建一個(gè)實(shí)例,調(diào)用 execute():
盡管比AsyncTask更復(fù)雜,你可能還是想創(chuàng)建自己的線程或者HandlerThread類,如果這么做,你應(yīng)該調(diào)用Process.setThreadPriority(THREAD_PRIORITY_BACKGROUND) 設(shè)置線程優(yōu)先線為”background”.如果沒有,線程仍然會(huì)拖慢應(yīng)用,因?yàn)樗鶸I線程優(yōu)先級(jí)相同。
如果你實(shí)現(xiàn)Thread或HandlerThread,確保UI線程沒有因?yàn)榈却ぷ骶€程執(zhí)行完而阻塞。不要調(diào)用Thread.wait()或Thread.sleep(),而是提供一個(gè)Handler,供任務(wù)執(zhí)行完后回調(diào)。如此設(shè)計(jì),UI線程會(huì)保持響應(yīng),避免出現(xiàn)ANR對(duì)話框。
特別強(qiáng)調(diào)BroadcastReceiver的執(zhí)行時(shí)間,意味著你要:分散工作到后臺(tái)線程里,像保存設(shè)置或者注冊(cè)Notification。執(zhí)行密集任務(wù)(intensive tasks),應(yīng)該用IntentService。
提示:你可以用StrictMode幫你找到在UI線程上潛在的耗時(shí)操作
相關(guān)文章
Android Handler內(nèi)存泄漏原因及解決方案
這篇文章主要介紹了Android Handler內(nèi)存泄漏原因及解決方案,幫助大家更好的理解和利用Android進(jìn)行開發(fā),感興趣的朋友可以了解下2021-02-02一文詳解?Compose?Navigation?的實(shí)現(xiàn)原理
這篇文章主要介紹了一文詳解?Compose?Navigation的實(shí)現(xiàn)原理,文章通告圍繞主題展開詳細(xì)的相關(guān)內(nèi)容介紹,具有一定的參考價(jià)值,需要的小伙伴可以參考一下2022-08-08Android仿硅谷商城實(shí)現(xiàn)購(gòu)物車實(shí)例代碼
這篇文章主要介紹了Android購(gòu)物車編輯實(shí)現(xiàn),小編覺得挺不錯(cuò)的,一起跟隨小編過來(lái)看看吧2018-05-05Android Studio如何快速導(dǎo)入jar和.so文件
這篇文章主要介紹了Android Studio如何快速導(dǎo)入jar和.so文件的相關(guān)知識(shí),非常不錯(cuò),具有參考借鑒價(jià)值,需要的朋友可以參考下2017-12-12Android仿微信Viewpager-Fragment惰性加載(lazy-loading)
這篇文章主要為大家詳細(xì)介紹了Android仿微信Viewpager-Fragment惰性加載lazy-loading,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-08-08Flutter?繪制風(fēng)車實(shí)現(xiàn)示例詳解
這篇文章主要為大家介紹了Flutter?繪制風(fēng)車實(shí)現(xiàn)示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-11-11