资讯专栏INFORMATION COLUMN

JVM如何捕获异常?

liukai90 / 2501人阅读

摘要:组成指针,指针代表捕获异常的范围,就是的范围。当触发异常的字节码的索引值在某个异常表条目的监控范围内,虚拟机会判断所抛出的异常和该条目想要捕获的异常是否匹配。

Java异常知识 1.异常的两大关键因素

(1)抛出异常
1.显式:应用程序手动抛出异常。具体就是使用throw抛出异常
2.隐式:Java虚拟机对于无法执行的代码,自动抛出异常
(2)捕获异常
1.try 代码块:用来标记需要进行异常监控的代码。
2.catch 代码块:跟在 try 代码块之后,用来捕获在 try 代码块中触发的某种指定类型的异常。除了声明所捕获异常的类型之外,catch 代码块还定义了针对该异常类型的异常处理器。在 Java中,try 代码块后面可以跟着多个 catch 代码块,来捕获不同类型的异常。Java 虚拟机会从上至下匹配异常处理器。因此,前面的 catch 代码块所捕获的异常类型不能覆盖后边的,否则编译器会报错。
3.fnally 代码块:跟在 try 代码块和 catch 代码块之后,用来声明一段必定运行的代码。它的设计初衷是为了避免跳过某些关键的清理代码,例如关闭已打开的系统资源。在程序正常执行的情况下,这段代码会在 try 代码块之后运行。否则,也就是 try 代码块触发异常的情况下,如果该异常没有被捕获,fnally 代码块会直接运行,并且在运行之后重新抛出该异常。如果该异常被 catch 代码块捕获,fnally 代码块则在 catch 代码块之后运行。在某些不幸的情况下,catch 代码块也触发了异常,那么 fnally 代码块同样会运行,并会抛出 catch 代码块触发的异常。在某些极端不幸的情况下,fnally 代码块也触发了异常,那么只好中断当前 fnally 代码块的执行,并往外抛异常。

2.异常的分类

1.所有异常的父类都是Throwable
2.Error异常是程序的执行状态无法恢复的状态,只能中止线程甚至中止JVM的异常
3.Exception是相对Error没有这么严重的异常
4.Runtime Exception和Error都属于不需要检查的异常
5.除了Runtime Exception和Error的异常都是Check Exception异常
6.Check Exception异常都是需要显式捕获的异常
3.Java虚拟机是如何捕获异常的?
java虚拟机构造异常实例非常昂贵。虚拟机需要生成该异常的栈轨迹。该操作会逐一访问当前线程的 Java 栈帧,并且记录下各种调试信息,包括栈帧所指向方法的名字,方法所在的类名、文件名,以及在代码中的第几行触发该异常。
既然异常实例的构造十分昂贵,我们是否可以缓存异常实例,在需要用到的时候直接抛出呢?从语法角度上来看,这是允许的。然而,该异常对应的栈轨迹并非 throw 语句的位置,而是新建异常的位置。
因此,这种做法可能会误导开发人员,使其定位到错误的位置。这也是为什么在实践中,我们往往选择抛出新建异常实例的原因。

异常处理器
1.来源:每个方法在编译的时候都会生成一个异常表。异常表里面的每一个条目都代表一个异常处理器。
2.组成:
(1)from指针,to指针:代表捕获异常的范围,就是Try的范围。
(2)target指针:代表处理器的开始位置,就是catch的起始位置。
(3)捕获的异常类型。
3.捕获异常
(1)当程序触发异常时,Java 虚拟机会从上至下遍历异常表中的所有条目。当触发异常的字节码的索引值在某个异常表条目的监控范围内,Java 虚拟机会判断所抛出的异常和该条目想要捕获的异常是否匹配。如果匹配,Java 虚拟机会将控制流转移至该条目 target 指针指向的字节码。
(2)如果遍历完所有异常表条目,Java 虚拟机仍未匹配到异常处理器,那么它会弹出当前方法对应的Java 栈帧,并且在调用者(caller)中重复上述操作。在最坏情况下,Java 虚拟机需要遍历当前线程 Java 栈上所有方法的异常表。
4.finally代码的编译:当前版本 Java 编译器的做法,是复制 fnally 代码块的内容,分别放在 try-catch 代码块所有正常执行路径以及异常执行路径的出口中。

代码1:
Try{
Try block
} catch {
Catch block
} finally {
Finally block
}
代码2:
Try {
Try block
Finally block
} catch {
Catch block
Finally block
} finally{
Finally block
}
代码1是我们的Java代码,代码2是编译之后的Java代码。

注意:如果 catch 代码块捕获了异常,并且触发了另一个异常,那么 fnally 捕获并且重抛的异常是哪个呢?答案是后者。也就是说原本的异常便会被忽略掉,这对于代码调试来说十分不利。

5.Java7的 Supressed 异常以及语法糖

针对上节说的会将catch的异常忽略掉,Java7引入了 Supressed 异常处理这个问题。但是使用起来还是很麻烦(没有感受,

文章版权归作者所有,未经允许请勿转载,若此文章存在违规行为,您可以联系管理员删除。

转载请注明本文地址:https://www.ucloud.cn/yun/73759.html

相关文章

  • 异常机制详解

    摘要:当运行时系统遍历调用栈而未找到合适的异常处理器,则运行时系统终止。不可查异常编译器不要求强制处置的异常包括运行时异常与其子类和错误。 目录介绍 1.什么是异常 2.异常 2.1 异常的概述和分类【了解】 2.2 JVM默认是如何处理异常的【理解】 2.3 异常处理的两种方式【理解】 2.4 try...catch的方式处理异常【掌握】 2.5 编译期异常和运行期异常的区别【理解】...

    wanghui 评论0 收藏0
  • Java异常处理

    摘要:可以被异常处理机制使用,是异常处理的核心。非检测异常,在编译时,不会提示和发现异常的存在,不强制要求程序员处理这样的异常。总体来说,语言的异常处理流程,从程序中获取异常信息。处理运行时异常,采用逻辑合理规避同时辅助处理。 目录 什么是Java异常? 当一个Exception在程序中发生的时候,JVM是怎么做的呢? 当我们编写程序的时候如何对待可能出现的异常呢? 正文 1. 什么是J...

    Fourierr 评论0 收藏0
  • 【JAVA系列】关于异常的一些事

    摘要:程序块若有异常发生,程序的运行便重点,并抛出异常类所产生的对象。关键字我们可以使用关键字把可能抛出的异常显式的标注在方法定义的位置从而提醒调用者要注意捕获这些异常。 ...

    不知名网友 评论0 收藏0
  • 聊聊Java的异常机制及实现

    摘要:是那些可能在虚拟机正常运行期间抛出的异常的超类。运行时异常定义及其子类都被称为运行时异常。对于语言中的关键字和,虚拟机中并没有特殊的字节码指令去支持它们,都是通过编译器生成字节码片段以及不同的异常处理器来实现。 前言 在一些传统的编程语言,如C语言中,并没有专门处理异常的机制,程序员通常用方法的特定返回值来表示异常情况,并且程序的正常流程和异常流程都采用同样的流程控制语句。Java语言...

    Towers 评论0 收藏0
  • Java异常简介

    摘要:而异常可以不被显式的处理都是的子类,继承了的就是异常,其他的就是异常。常见异常类列举几个常见的运行时异常数组越界异常空指针异常类转换异常数字格式异常运算异常。 Java异常 java异常分为两大类,Checked异常和Runtime异常,Checked异常都是在编译阶段可以被处理的异常。 Checked异常和Runtime异常的区别和联系 Checked异常都是可以被处理的异常,在程...

    wangym 评论0 收藏0

发表评论

0条评论

liukai90

|高级讲师

TA的文章

阅读更多
最新活动
阅读需要支付1元查看
<