Java和Android崩潰捕獲機(jī)制
引言
作為開發(fā)同學(xué),每天都在面臨各種各種的崩潰問題。
我們都如果在Android應(yīng)用中發(fā)生了未捕獲的崩潰問題,不管是在主線程還是在子線程,應(yīng)用都會(huì)直接退出。
但是Java程序,子線程拋出的異常,不會(huì)引起程序的退出。
那你們知道JVM是如何處理應(yīng)用未捕獲崩潰的嗎?Android又是怎樣在發(fā)生崩潰時(shí)讓程序退出的呢?
崩潰處理機(jī)制
當(dāng)一個(gè)線程拋出異常時(shí),JVM會(huì)調(diào)用線程的dispatchUncaughtException方法,所有未被捕獲的異常,最后都會(huì)交給UncaughtExceptionHandler處理。
對于一個(gè)線程來說,UncaughtExceptionHandler有多個(gè),首先有針對單個(gè)線程的unCaughtExceptionHandler,然后還有靜態(tài)的首先有一個(gè)靜態(tài)的defaultUncaughtExceptionHandler和defaultUncaughtPreExceptionHandler,這個(gè)是對每個(gè)線程都生效的。
處理順序:未捕獲的異常,先由線程處理,然后由線程的ThreadGroup處理,最后再由默認(rèn)異常處理程序處理。
Android發(fā)生崩潰后
為什么Android發(fā)生異常后,不管是在主線程還是在子線程,都會(huì)引起程序crash退出呢?
其實(shí)是因?yàn)锳ndroid給所有線程都設(shè)置了一個(gè)defaultExceptionHandler,這個(gè)ExceptionHandler的處理邏輯就是讓程序退出。
下面我們來看源碼。
在應(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的原理來實(shí)現(xiàn)Android應(yīng)用永不崩潰。
以上就是Java和Android崩潰捕獲機(jī)制的詳細(xì)內(nèi)容,更多關(guān)于Java Android崩潰捕獲的資料請關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
spring中@Autowire和@Resource的區(qū)別在哪里(推薦)
這篇文章主要介紹了spring中@Autowire和@Resource的區(qū)別在哪里?本文結(jié)合實(shí)例代碼給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2023-02-02
java開發(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
java如何對map進(jìn)行排序詳解(map集合的使用)
這篇文章主要介紹了java如何對map進(jìn)行排序,java map集合的使用詳解,大家可以參考使用2013-12-12
Java實(shí)現(xiàn)隨機(jī)生成大小寫混合的卡密的步驟
在現(xiàn)代軟件開發(fā)中,生成隨機(jī)卡密是一個(gè)常見的需求,尤其是在需要為用戶生成唯一識(shí)別碼或安全令牌的場景中,卡密通常由數(shù)字和字母組成,有時(shí)還會(huì)包含特殊字符,本文通過代碼講解的非常詳細(xì),需要的朋友可以參考下2024-11-11

