c语言sscanf函数的用法是什么
269
2023-07-12
JVM处理未捕获异常的方法详解
前言
继之前的文章详解JVM如何处理异常,今天再次发布一篇比较关联的文章,如题目可知,今天聊一聊在JVM中线程遇到未捕获异常的问题,其中涉及到线程如何处理未捕http://获异常和一些内容介绍。
什么是未捕获异常
未捕获异常指的是我们在方法体中没有使用try-catch捕获的异常,比如下面的例子
private static void testUncaughtException(String arg) {
try {
System.out.println(1 / arg.length());
} catch (ArithmeticException e) {
e.printStackTrace();
}
}
上面的代码很有可能发生如下情况
如果方法参数arg传递null,会出现NullPointerException
如果参数arg传递内容为空的字符串(“”),会出现ArithmeticException
对于上面的问题,我们不难发现
上面可能出现的NullPointerException和ArithmeticException都属于Unchecked Exceptions
而ArithmeticException被我们人为try-catch捕获了,它不符合本文对于未捕获异常的定义
NullPointerException 由于我们没有catch住,就变成了我们要聊的未捕获异常
另外,未捕获异常实际是Unchecked Exceptions的子集
UncaughtExceptionHandler 是什么
它是线程遇到未捕获异常的一个处理者接口
它包含一个方法void uncaughtException(Thread t, Throwable e); 用来处理接收处理异常发生后的操作,比如收集崩溃信息并上报等
可以通过 实例方法 Thread.setUncaughtExceptionHandler 为某一个Thread实例设置未捕获异常处理者
也可以通过 静态方法 Thread.setDefaultUncaughtExceptionHandler 设置所有Thread实例的未捕获异常处理者
ThreadGroup 是什么
ThreadGroup 是线程的集合
ThreadGroup 也可以包含子ThreadGroup
除了初始的ThreadGroup 之外,每个ThreadGroup都有一个父 ThreadGroup
ThreadGroup 自身实现了Thread.UncaughtExceptionHandler,用来相应处理其内部的线程和ThreadGroup发生未捕获异常。
未捕获异常处理者 设置指南
线程发生了未捕获异常,JVM怎么处理
分发Throwable实例
当线程A中出现了未捕获异常时,JVM会调用线程A的dispatchUncaughtException(Throwable)方法
/**
* DAdQAIkRJispatch an uncaught exception to the handler. This method is
* intended to be called only by the JVM.
*/
private void dispatchUncaughtException(Throwable e) {
getUncaughtExceptionHandler().uncaughtException(this, e);
}
获取未捕获异常处理者
每个线程会有一个变量(uncaughtExceptionHandler)来保存未捕获异常的处理者
在线程需要确定Throwable分发目标的处理者时,优先获取当前线程中uncaughtExceptionHandler变量
如果出问题线程的uncaughtExceptionHandler为null(即没有显式设置异常处理者),则使用自己所在的ThreadGroup来作为未捕获异常处理者。
/**
* Returns the handler invoked when this thread abruptly terminates
* due to an uncaught exception. If this thread has not had an
* uncaught exception handler explicitly set then this thread's
* ThreadGroup object is returned, unless this thread
* has terminated, in which case null is returned.
* @since 1.5
* @return the uncaught exception handler for this thread
*/
public UncaughtExceptionHandler getUncaughtExceptionHandler() {
return uncaughtExceptionHandler != null ?
uncaughtExceptionHandler : group;
}
如果Throwable分发给ThreadGroup
ThreadGroup会尝试转给它的父ThreadGroup(如果存在的话)
如果上面没有找到对应的ThreadGroup,则尝试获取Thread.getDefaultUncaughtExceptionHandler()并分发
/**
* Called by the java Virtual Machine when a thread in this
* thread group stops because of an uncaught exception, and the thread
* does not have a specific {@link Thread.UncaughtExceptionHandler}
* installed.
*
* The uncaughtException
method of
* ThreadGroup
does the following:
*
*
* uncaughtException
method of that parent is called
* with the same two arguments.
*
* {@linkplain Thread#getDefaultUncaughtExceptionHandler default
* uncaught exception handler} installed, and if so, its
* uncaughtException
method is called with the same
* two arguments.
*
* argument is an instance of {@link ThreadDeath}. If so, nothing
* special is done. Otherwise, a message containing the
* thread's name, as returned from the thread's {@link
* Thread#getName getName} method, and a stack backtrace,
* using the Throwable
's {@link
* Throwable#printStackTrace printStackTrace} method, is
* printed to the {@linkplain System#err standard error stream}.
*
*
* Applications can override this method in subclasses of
* ThreadGroup
to provide alternative handling of
* uncaught exceptions.
*
* @param t the thread that is about to exit.
* @param e the uncaught exception.
* @since JDK1.0
*/
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);
}
}
}
将上面的处理流程做成图的形式,就是下图所示
注:上述图片来自https://javamex.com/tutorials/exceptions/exceptions_uncaught_handler.shtml
Questions
初始的ThreadGroup是什么
上面提到了初始的ThreadGroup没有父ThreadGroup,是主线程所在的ThreadGroup么?
这个问题,我们可以通过这样一段代码验证
private static void dumpThreadGroups() {
ThreadGroup threadGroup = Thread.currentThread().getThreadGroup();
while(threadGroup != null) {
System.out.println("dumpThreadGroups threadGroup=" + threadGroup.getName());
threadGroup = threadGroup.getParent();
}
}
执行该AdQAIkRJ方法对应的输出是
dumpThreadGroups threadGroup=main
dumpThreadGroups threadGroup=system
因此我们可以发现,初始的ThreadGroup是一个叫做system的ThreadGroup,而不是main ThreadGroup
setDefaultUncaughtExceptionHandler 设置的一定会被调用到么
这其实是一个很好的问题,答案是不一定会被调用,因为可能存在以下的情况
出问题的线程设置了对应的UncaughtExcpetionHandler,优先响应分发到这个Handler
出问题的线程所在的ThreadGroup包括其祖先ThreadGroup 重写了uncaughtException 也可能造成线程默认的Handler无法被调用
出问题的线程重写了dispatchUncaughtException 可能性较小
出问题的线程重写了getUncaughtExceptionHandler 可能性较小
参考声明
How uncaught exceptions are handled
总结
以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,如果有疑问大家可以留言交流,谢谢大家对我们的支持。
版权声明:本文内容由网络用户投稿,版权归原作者所有,本站不拥有其著作权,亦不承担相应法律责任。如果您发现本站中有涉嫌抄袭或描述失实的内容,请联系我们jiasou666@gmail.com 处理,核实后本网站将在24小时内删除侵权内容。
发表评论
暂时没有评论,来抢沙发吧~