Java和Android崩潰捕獲機(jī)制
引言
作為開(kāi)發(fā)同學(xué),每天都在面臨各種各種的崩潰問(wèn)題。
我們都如果在Android應(yīng)用中發(fā)生了未捕獲的崩潰問(wèn)題,不管是在主線程還是在子線程,應(yīng)用都會(huì)直接退出。
但是Java程序,子線程拋出的異常,不會(huì)引起程序的退出。
那你們知道JVM是如何處理應(yīng)用未捕獲崩潰的嗎?Android又是怎樣在發(fā)生崩潰時(shí)讓程序退出的呢?
崩潰處理機(jī)制
當(dāng)一個(gè)線程拋出異常時(shí),JVM會(huì)調(diào)用線程的dispatchUncaughtException
方法,所有未被捕獲的異常,最后都會(huì)交給UncaughtExceptionHandler處理。
對(duì)于一個(gè)線程來(lái)說(shuō),UncaughtExceptionHandler
有多個(gè),首先有針對(duì)單個(gè)線程的unCaughtExceptionHandler
,然后還有靜態(tài)的首先有一個(gè)靜態(tài)的defaultUncaughtExceptionHandler
和defaultUncaughtPreExceptionHandler
,這個(gè)是對(duì)每個(gè)線程都生效的。
處理順序:未捕獲的異常,先由線程處理,然后由線程的ThreadGroup
處理,最后再由默認(rèn)異常處理程序處理。
Android發(fā)生崩潰后
為什么Android發(fā)生異常后,不管是在主線程還是在子線程,都會(huì)引起程序crash退出呢?
其實(shí)是因?yàn)锳ndroid給所有線程都設(shè)置了一個(gè)defaultExceptionHandler
,這個(gè)ExceptionHandler
的處理邏輯就是讓程序退出。
下面我們來(lái)看源碼。
在應(yīng)用程序被創(chuàng)建的時(shí)候,RuntimeInit
會(huì)設(shè)置一個(gè)默認(rèn)的異常處理Handler
,這個(gè)異常處理Handler
就是KillApplicationHandler
。從名字就可以看出,這個(gè)Handler
主要負(fù)責(zé)殺掉App進(jìn)程。
// RuntimInit protected static final void commonInit() { LoggingHandler loggingHandler = new LoggingHandler(); // 設(shè)置preExceptionHandler Thread.setUncaughtExceptionPreHandler(loggingHandler); // KillApplicationHandler 作為全局 Handler Thread.setDefaultUncaughtExceptionHandler(new KillApplicationHandler(loggingHandler)); //... }
KillApplicationHandler
會(huì)先調(diào)用loggingHandler
打印日志,然后殺掉當(dāng)前進(jìn)程。
private static class KillApplicationHandler implements Thread.UncaughtExceptionHandler { private final LoggingHandler mLoggingHandler; public KillApplicationHandler(LoggingHandler loggingHandler) { // 傳入loggingHandler用于打日志 this.mLoggingHandler = Objects.requireNonNull(loggingHandler); } @Override public void uncaughtException(Thread t, Throwable e) { try { // 打日志 ensureLogging(t, e); // 已經(jīng)在crash中了,不處理了 if (mCrashing) return; mCrashing = true; // ... } catch (Throwable t2) { // ... } finally { // 通知內(nèi)核殺掉進(jìn)程 Process.killProcess(Process.myPid()); // 停止VM System.exit(10); } }
所以,當(dāng)出現(xiàn)未捕獲的異常時(shí),會(huì)交給KillApplicationHandler
中的uncaughtException
,從而直接讓程序退出。與此同時(shí),我們也可以從adb日志中看到崩潰的具體堆棧。
下一篇,我們講講如何借用 uncaughtExceptionHandler
的原理來(lái)實(shí)現(xiàn)Android應(yīng)用永不崩潰。
以上就是Java和Android崩潰捕獲機(jī)制的詳細(xì)內(nèi)容,更多關(guān)于Java Android崩潰捕獲的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
spring中@Autowire和@Resource的區(qū)別在哪里(推薦)
這篇文章主要介紹了spring中@Autowire和@Resource的區(qū)別在哪里?本文結(jié)合實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2023-02-02java開(kāi)發(fā)web前端cookie session及token會(huì)話機(jī)制詳解
如果把人體比作一個(gè)web系統(tǒng)的話,cookie、session和token就好像人體的經(jīng)絡(luò)和血管一樣,而web系統(tǒng)中的數(shù)據(jù),就好像人體的血液一樣。血液依靠著血管在人體內(nèi)流動(dòng),就如數(shù)據(jù)根據(jù)cookie和session機(jī)制在web系統(tǒng)中流動(dòng)一樣2021-10-10微信支付之公眾號(hào)支付(java實(shí)現(xiàn))
這篇文章主要介紹了微信支付之公眾號(hào)支付(java實(shí)現(xiàn)),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2019-10-10java如何對(duì)map進(jìn)行排序詳解(map集合的使用)
這篇文章主要介紹了java如何對(duì)map進(jìn)行排序,java map集合的使用詳解,大家可以參考使用2013-12-12Java實(shí)現(xiàn)隨機(jī)生成大小寫混合的卡密的步驟
在現(xiàn)代軟件開(kāi)發(fā)中,生成隨機(jī)卡密是一個(gè)常見(jiàn)的需求,尤其是在需要為用戶生成唯一識(shí)別碼或安全令牌的場(chǎng)景中,卡密通常由數(shù)字和字母組成,有時(shí)還會(huì)包含特殊字符,本文通過(guò)代碼講解的非常詳細(xì),需要的朋友可以參考下2024-11-11