第17章 异常

javac为remainder ()方法产生下列字节码序列:

// The main bytecode sequence for remainder ();
// Push local variable 0 {arg passed as
0 iload_0 // dividend)
// Push the minimum integer value
1 ldc #1 <Integer -2147483648>

// If the dividend isn't equal to the minimum
// integer, jump to the remainder calculation
3 if_icmpne 19

// Push local variable 1 (arg pased as

6 iload_l // divisor)

// Push -1

7 iconst_ml

// If the divisor isn't equal to -1, jump
// to the remainder calculation

8 if_icmpne 19

// This is an overflow case, bo throw an
// exception. Create a new OverflowException,
// push reference to it onto the stack

11 new #4 <ClaBS OverflowException>
14 dup //make a copy of the reference

//Pop one copy of the reference and invoke
// the <init> method of new OverflowException
// object

15 invokeapecial #10 <Method OverflowException()>

// Pop the other reference to the
// OverflowException and throw it

18 athrow

// Calculate the remainder
// Puab local variable 0 (arg passed as
19 iload_0 // dividend)
// Push local variable 1 (arg passed as
20 iload_l // divisor)
// Pop divisor; pop dividend; calculate,push
21 irem // remainder
22 ireturn // Return int on top of stack (the remainder)

// The bytecode sequence for the
// catch (ArithmeticBxception) clause:

// Pop the reference to the
// ArithmoticException because it isn't used

23 pop //by this catch clause.

24 new #2 <Class DivideByZeroException>
// Create and push reference to new object of
// class DivideByZeroException.
// Duplicate the reference to the new object
// on the top of the stack because it must be
// both initialized and thrown. The
// initialization will consume the copy of the

27 dup // reference created by the dup,

// Call the no-arg <init> method for the
// DivideByZeroException to initialize it.

// This instruction will pop the top reference
// to the object.

28 invokespecial #9 <Method DivideByZeroException()>

// Fop the reference to a Throwable object, in
// this case the DivideByZeroException,

31 athrow // and throw the exception.
方法remainder ()的字节码序列包含两个部分:第一部分是方法的正常执行路径,该部分指的是pc指针偏移量0到22;第二部分是catch子句,该部分指的是pc指针偏移量23到31。

// Overflow can occur in division when dividing
// the negative integer of the largest possible
// magnitude (Integer.MIN_VALUE) by -1, because
// this would Juat flip the sign, but there is no
// way to represent that nunber in an int.
if((dividend Integer.MIN_VALUE> &&(divisor==-1)){
throw new OverflowException();
}
所有作为Error和RuntimeException的子类的异常都是未检验的。

17.2异常表
主字节码序列中的指令irem可能会抛出ArithmeticException异常。如果发生这种情况,Java 虚拟机将在表中査找异常,然后跳转到实现catch子句的字节码序列。每个捕获异常的方法都与异常表相关联,该异常表与方法的字节码序列一起送到Class文件中。每一个被try语句块捕获的异常都与异常表中的一个入口(项)相对应。异常表中的每个入口都包括四部分信息:

•起点。

•终点。

•将要跳转到的字节码序列中的pc指针偏移量。

•被捕获的异常类的常量池索引。

下面是NitPickyMath类的remainder ()方法的异常表:

Exception table:

from to target type

I9 23 23 <Class java.lang.ArithmeticException>

上述异常表说明,ArithmeticException异常在pc指针偏移量19到22 (含)中被捕获。try语句块的终点值在“to”栏中列出。这个终点值总是比捕获异常位置的pc指针偏移量的最大值还要大1。在这个例子中,终点值为23,但捕获异常位置的pc偏移量最大值为22。偏移量19到22 (含)之间的语句对应于实现remainder ()内try语句块代码的字节码序列。上述表格中列出的target栏指出了如果ArithmeticException异常在pc指针偏移量19到22 (含)之间抛出,pc指针偏移量将要跳转到的位置。

如果异常在方法执行时抛出,Java虚拟机将会在整个异常表中搜寻相匹配的项。如果当前程序计数器在异常表入口所指定的范围内,而且所抛出的异常类是该入口所指向的类(或为指定类的子类),那么该入口即为所搜寻的入口。Java虚拟机按照每个入口在表中出现的顺序进行检索。当遇到第一个匹配项时,虚拟机将程序计数器设为新的pc指针偏移量位置,然后从该位置 继续执行。如果没有发现相匹配的项,虚拟机将当前栈帧从栈中弹出,再次抛出同样的异常。 当Java虚拟机弹出当前栈帧时,虚拟机马上终止当前方法的执行,并且返回至调用本方法的方法 中,但是并非继续正常执行该方法,而是在该方法中拋出同样的异常,这就使得虚拟机在该方 法中再次执行同样的搜寻异常表的操作。

Java程序员可以使用throw语句抛出异常,正如remainder ()方法中的catch ( Arithmetic Exception)子句,这里创建了异常DivideByZcjroExceplion,并将其抛出。执行抛出异常操作的 字节码如表17-1所示。

指令athrow从栈顶端弹出宇,并假设它是一个对象的引用,该对象为Throwable类的子类的实例(或者就是Throwable类的实例)。所抛出异常的类型由弹出的对象引用类型决定。

posted @ 2019-12-03 22:41  mongotea  阅读(121)  评论(0编辑  收藏  举报