[Android-Framework] Crash for Java,程序员的中年危机

本文探讨了Android应用程序中Java崩溃捕获的过程,包括注册、分发和处理异常的步骤。详细阐述了如何通过Thread的UncaughtExceptionHandler进行日志记录和异常处理,以及Android系统如何默认处理未捕获的异常,确保App的稳定运行。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >


从安卓 APP 开发的角度,Java 崩溃捕获相对比较容易,虚拟机给 Java 字节码提供了一个受控的运行环境,同时也提供了完善的 Java 崩溃捕获机制。

2.1 注册

Thread中定义了Thread.UncaughtExceptionHandler 接口。

public interface UncaughtExceptionHandler {

void uncaughtException(Thread t, Throwable e);
}

Thread中存在两个UncaughtExceptionHandler字段,一个是静态的,另一个是非静态

静态:defaultUncaughtExceptionHandler
静态方法:public static void setDefaultUncaughtExceptionHandler()
当前进程中的任何线程抛出了未捕获Exception,都会被此UncaughtExceptionHandler处理,因此管辖范围为整个进程。

非静态:uncaughtExceptionHandler
实例方法:public void setUncaughtExceptionHandler(),即线程私有的。
为单个线程设置一个属于线程自己的uncaughtExceptionHandler,管辖范围仅限于自己的线程。

扩展:App在发生Crash的时候,是如何打印日志,如何退出的?
App启动时,Android会通过zygote进程fork一个进程,然后执行初始化工作。在初始化的过程中,会执行RuntimeInit.commonInit();中的方法:

// /frameworks/base/core/java/com/android/internal/os/RuntimeInit.java

protected static final void commonInit() {
if (DEBUG) Slog.d(TAG, “Entered RuntimeInit!”);

/*

  • set handlers; these apply to all threads in the VM. Apps can replace
  • the default handler, but not the pre handler.
    */
    LoggingHandler loggingHandler = new LoggingHandler();
    Thread.setUncaughtExceptionPreHandler(loggingHandler);
    Thread.setDefaultUncaughtExceptionHandler(new KillApplicationHandler(loggingHandler));
    // …略
    }

在上面的代码中我们可以发现,Android已经为App设置了默认的UncaughtExceptionHandler进行异常处理:

  • LoggingHandler:用于处理打印日志。
  • KillApplicationHandler:调用AMS,弹出“应用异常”的dialog,然后结束进程并退出等等。

2.2 分发

当发生未捕获的异常(没有被try-catch的异常)时,虚拟机会调用当前线程Thread的dispatchUncaughtException方法,进行异常分发。

// art/runtime/thread.cc

void Thread::HandleUncaughtExceptions(ScopedObjectAccessAlreadyRunnable& soa) {
if (!IsExceptionPending()) {
return;
}
ScopedLocalRef peer(tlsPtr_.jni_env, soa.AddLocalReference(tlsPtr_.opeer));
ScopedThreadStateChange tsc(this, kNative);

// Get and clear the exception.
ScopedLocalRef exception(tlsPtr_.jni_env, tlsPtr_.jni_env->ExceptionOccurred());
tlsPtr_.jni_env->ExceptionClear();

// 调用当前线程的dispatchUncaughtException方法进行分发
// Call the Thread instance’s dispatchUncaughtException(Throwable)
tlsPtr_.jni_env->CallVoidMethod(peer.get(),
WellKnownClasses::java_lang_Thread_dispatchUncaughtException,
exception.get());

// If the dispatchUncaughtException threw, clear that exception too.
tlsPtr_.jni_env->ExceptionClear();
}

所以,Thread的dispatchUncaughtException负责处理异常的分发逻辑。

2.2.1 分发过程

  1. 首先处理setUncaughtExceptionPreHandler()方法注册的异常处理Handler —— 这里的是LoggingHandler,负责记录日志(默认,不可修改过)

// java/lang/Thread.java

public final void dispatchUncaughtException(Throwable e) {
// 上面进行打印日志
// BEGIN Android-added: uncaughtExceptionPreHandler for use by platform.
Thread.UncaughtExceptionHandler initialUeh =
Thread.getUncaughtExceptionPreHandler();
if (initialUeh != null) {
try {
initialUeh.uncaughtException(this, e);
} catch (RuntimeException | Error ignored) {
// Throwables thrown by the initial handler are ignored
}
}
// END Android-added: uncaughtExceptionPreHandler for use by platform.

// 然后在这里进行分发和处理
getUncaughtExceptionHandler().uncaughtException(this, e);
}

  1. 然后处理线程私有的uncaughtExceptionHandler异常处理方法;如果私有Handler不存在,则使用ThreadGroup的异常处理方法;

// java/lang/Thread.java

public UncaughtExceptionHandler getUncaughtExceptionHandler() {
return uncaughtExceptionHandler != null ?
uncaughtExceptionHandler : group;
}

  1. ThreadGroup中,优先调用父线程组的处理逻辑,否则,调用setDefaultUncaughtExceptionHandler() 方法注册的Handler处理异常。

// java/lang/ThreadGroup.java

public void uncaughtException(Thread t, Throwable e) {
if (parent != null) {
parent.uncaughtException(t, e);
} else {
Thread.UncaughtExceptionHandler ueh = Thread.getDefaultUncaughtExceptionHandler();
if (ueh != null) {
ueh.uncaughtException(t, e);
} else if (!(e instanceof ThreadDeath)) {
System.err.print("Exception in thread “” + t.getName() + “” ");
e.printStackTrace(System.err);
}
}
}

流程图如下: 2021-04-17-15-29-01.png

2.3 处理

Android已经为App设置了默认的UncaughtExceptionHandler进行异常处理:

  • LoggingHandler:用于打印日志。
  • KillApplicationHandler:调用AMS,弹出“应用异常”的dialog,结束进程并退出等等。

关键代码如下:

// /frameworks/base/core/java/com/android/internal/os/RuntimeInit.java

总结

【Android 详细知识点思维脑图(技能树)】

image

其实Android开发的知识点就那么多,面试问来问去还是那么点东西。所以面试没有其他的诀窍,只看你对这些知识点准备的充分程度。so,出去面试时先看看自己复习到了哪个阶段就好。

虽然 Android 没有前几年火热了,已经过去了会四大组件就能找到高薪职位的时代了。这只能说明 Android 中级以下的岗位饱和了,现在高级工程师还是比较缺少的,很多高级职位给的薪资真的特别高(钱多也不一定能找到合适的),所以努力让自己成为高级工程师才是最重要的。

这里附上上述的面试题相关的几十套字节跳动,京东,小米,腾讯、头条、阿里、美团等公司19年的面试题。把技术点整理成了视频和PDF(实际上比预期多花了不少精力),包含知识脉络 + 诸多细节。

由于篇幅有限,这里以图片的形式给大家展示一小部分。

详细整理在GitHub:Android架构视频+BAT面试专题PDF+学习笔记​

id-P7/blob/master/Android%E5%BC%80%E5%8F%91%E4%B8%8D%E4%BC%9A%E8%BF%99%E4%BA%9B%EF%BC%9F%E5%A6%82%E4%BD%95%E9%9D%A2%E8%AF%95%E6%8B%BF%E9%AB%98%E8%96%AA%EF%BC%81.md)**

网上学习 Android的资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。希望这份系统化的技术体系对大家有一个方向参考。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值