Java虚拟机通过编译期生成的异常表定位异常处理器,表项含start_pc、end_pc、handler_pc和catch_type四字段;JVM按顺序匹配当前PC是否在区间内且异常类型兼容,找到首个匹配项即跳转执行。
Java虚拟机通过方法的异常表(Exception Table)来定位和处理字节码中的异常跳转,而不是靠逐行扫描或运行时动态解析。异常表是编译期生成、存于Code属性中的一组结构化规则,JVM在抛出异常时按顺序匹配表项,找到第一个覆盖当前异常位置且匹配异常类型的处理器,然后跳转到其指定的处理代码(即handler_pc)。
每个异常表项包含四个 2 字节无符号整数字段:
Class_info,表示该 handler 能捕获的异常类型;值为 0 表示 finally 或 try-with-resources 的 finally 部分(不依赖异常类型)当某条字节码指令执行中抛出异常时,JVM按以下逻辑查找处理器:
pc
start_pc ≤ pc ,且异常实例是 catch_type 所指类或其子类(若 catch_type ≠ 0)
handler_pc,继续执行编译器会为不同语义生成多个异常表项:
catch 块对应一项,catch_type 指向对应异常类,start_pc/end_pc 覆盖整个 try 区域finally 块(包括隐式生成的)通常对应多项:一项用于正常流程(catch_type = 0,覆盖 try 和所有 catch),若干项用于异常流程(同样 catch_type = 0,但可能覆盖不同区间,确保 finally 总被执行)可通过 javap -v 查看编译后 class 文件的异常表:
javap -v MyClass | grep -A2
0 "Exception table"
from(start_pc)、to(end_pc)、target(handler_pc)、type(类名或 any 表示 catch_type=0)基本上就这些。异常表机制轻量、确定性强,是 JVM 实现高效异常处理的基础,不依赖反射或运行时类型推导,所有匹配都在常量时间内完成。