一 .两种异常结构
java中的异常和错误都继承自java.lang.Throwable在异常处理的时候,都会接触到受检异常(checked exception)和非受检异常(unchecked exception)这两种异常类型。
非受检异常
指的是java.lang.RuntimeException和java.lang.Error类及其子类,所有其他的异常类都称为受检异常
。两种类型的异常在作用上并没有差别。
两者的区别主要在:受检的异常是由编译器强制执行
的,必须捕获,用于指示不受程序控制的异常情况(例如,I/O 错误),而非受检的异常在运行时发生,用于指示编程错误
(例如,空指针。正因为如此,受检异常在使用的时候需要比非受检异常更多的代码来避免编译错误。
二 .两者的常见异常
uncheckedExcepiton(RuntimeException) | CheckedException |
---|---|
Java.lang.ArithmeticException Java.lang.ArrayStoreExcetpion Java.lang.ClassCastException Java.lang.EnumConstantNotPresentException Java.lang.IllegalArgumentException Java.lang.IllegalThreadStateException Java.lang.NumberFormatException Java.lang.IllegalMonitorStateException Java.lang.IllegalStateException Java.lang.IndexOutOfBoundsException Java.lang.ArrayIndexOutOfBoundsException Java.lang.StringIndexOutOfBoundsException Java.lang.NegativeArraySizeException’ Java.lang.NullPointerException Java.lang.SecurityException Java.lang.TypeNotPresentException Java.lang.UnsupprotedOperationException | Java.lang.ClassNotFoundException Java.lang.CloneNotSupportedException Java.lang.IllegalAccessException Java.lang.InterruptedException Java.lang.NoSuchFieldException Java.lang.NoSuchMetodException |
非受检异常
RuntimeException在默认情况下会得到自动处理。所以通常用不着捕获RuntimeException,但在自己的封装里,也许仍然要选择抛出一部分RuntimeException。
RuntimeException
是那些可能在 Java 虚拟机正常运行期间抛出的异常的超类。可能在执行方法期间抛出但未被捕获的RuntimeException
的任何子类都无需在throws
子句中进行声明。(java api)-
受检异常,是值需要显示通过
Catch
捕获的异常,在Java中,除了RuntimeException以外的异常,都属于受检异常(checkedException).
我们以NoSuchMethodException
为例,如图所示,可以明显看到,该异常在没有捕获的情况下,会显示提示语法错误,有两个解决办法-
Add exception to method signature
,表示把这个异常再往上抛。 -
Surround with try/catch
,表示使用try/catch
捕获。
-
三 异常的选择
一直以来,关于在程序中到底是该使用受检异常还是非受检
我们通常需要保证程序不会捕捉到不在我们预期范围内的异常,比如RuntimeException,我们希望这类异常是要往外抛,而不是在内部被捕获。不要让它把异常吞掉,因为一旦程序出现问题,没有异常信息很难定位。
如果希望调用者能够从异常中进行合理恢复,需要设置为受检异常类型,如果调用者无法采用任何措施使得程序无法重异常中恢复,需要把该异常设置为非受检异常。
四. 扩展:一道经典的面试题
一道非常经典的面试题,NoClassDefFoundError 和 ClassNotFoundException 有什么区别?
-
NoClassDefFoundError,表示这个类在编译时期存在,但是在运行时不能找到合适的类导致的错误。例如在运行时我们想调用某个类的方法或者访问这个类的静态成员的时候,发现这个类不可用,此时Java虚拟机就会抛出NoClassDefFoundError错误。
可能出现的错误情况如下:
对应的Class在java的classpath中不可用
你可能用jar命令运行你的程序,但类并没有在jar文件的manifest文件中的classpath属性中定义
可能程序的启动脚本覆盖了原来的classpath环境变量
因为NoClassDefFoundError是java.lang.LinkageError的一个子类,所以可能由于程序依赖的原生的类库不可用而导致
检查日志文件中是否有java.lang.ExceptionInInitializerError这样的错误,NoClassDefFoundError有可能是由于静态初始化失败导致的
如果你工作在J2EE的环境,有多个不同的类加载器,也可能导致NoClassDefFoundError
- ClassNotFoundException,它是程序运行期间的异常,比如当我们尝试在运行时使用反射加载类时,ClassNotFoundException 就会出现。
@CallerSensitivepublic static Class<?> forName(String className) throws ClassNotFoundException
{
Class<?> caller = Reflection.getCallerClass();
return forName0(className, true, ClassLoader.getClassLoader(caller), caller);
}
总的来说,ClassNotFoundException 和 NoClassDefFoundError 都是由 CLASSPATH
中缺少类引起的,通常是由于缺少 JAR 文件而引起的,但是如果 JVM 认为应用运行时找不到相应的引用,就会抛出 NoClassDefFoundError 错误;当你在代码中显示的加载类比如 Class.forName() 调用时却没有找到相应的类,就会抛出java.lang.ClassNotFoundException
。
问题解答
面试题:请你说一下对受检异常和非受检异常的理解·
回答: 受检异常和非受检异常,都是派生自Throwable
这个类。他们的区别是
受检异常: 是指需要调用者显示通过try-catch捕获的异常
非受检异常: 是指不需要调用者显示捕获的异常。
之所以要定义受检异常和非受检异常主要是因为两者有着不同的作用
在程序中,存在一些需要用户在编译期间就去检查的问题,比如FileNotFoundException、IOException,这些异常涉及资源处理,调用者需要捕获,其实它可以提醒开发者,如果被调用的方法出现这类异常时,程序应该做好预判并处理,比如IOExcetion,我们需要对流进行关闭操作。
而非受检发生在运行期间,是程序运行过程中可能发生的错误类型,比如NullpointExcetpion,这些异常我们可以捕获,也可以不捕获。但是捕获这些异常只能打印一些日志,除此之外什么都做不了