前言
Java异常是Java提供的用于处理程序中错误的一种机制。异常处理机制能让程序在异常发生时,按照代码的预先设定的异常处理逻辑,针对性的处理异常,让程序尽最大可能恢复正常并继续执行,且保持代码的清晰。
异常的分类
Error:一般是环境问题,并非程序的问题。最典型的就是虚拟机中运行的程序过多,造成的虚拟机内存溢出,虚拟机错误,还有系统崩溃之类的,应用程序是无法处理此类错误的。Error不在我们调试代码可以处理的范围中。
Exception:这是我们需要关心的异常,因为当这些异常抛出时,说明了我们程序出现了问题,我们需要处理这些问题。exception又分为了不检查异常和检查异常。
不检查异常
Error和runtimeException都属于不检查异常。
public void method(String s) {
if(s == null)
throw new NullPointerException();
}
首先,我们在代码中不会写这样的代码,因为这样的情况在我们的代码中可能数不胜数,如果都需要我们去手动的抛出异常,那么我们的代码会很难维护,那么当出现使用了指向空的引用时,系统会自动帮你抛出,所以我们不用手动抛出所有RuntimeException类型(它的子类)的异常。
其次,当有方法会抛出RuntimeException类型的异常时(系统自动抛出),我们不需要去捕获它(try-catch),所以这类异常被称为不检查异常
。有经验的大家会知道,往往我们的程序的问题就是出现在这些不被检查的异常,如NullPointerException(经常在一些群中发现新手贴出空指针异常,然后求助……)。所以正因为RuntimeException类型的异常不用我们去捕获,因此我们在Code的时候一定要注意防止RuntimeException的发生。
检查异常
这一类异常抛出时,我们必须去捕获它,在方法中要么用try-catch语句捕获它并处理,要么用throws子句声明抛出它,否则编译不会通过。这样的异常一般是由程序的运行环境导致的。因为程序可能被运行在各种未知的环境下,而程序员无法干预用户如何使用他编写的程序,于是程序员就应该为这样的异常时刻准备着。如SQLException , IOException,ClassNotFoundException 等。至于说如何处理,根据具体的业务逻辑来编写处理的代码,那么接下来我们来看看Java中是如何设计异常处理的。
异常处理
在编写代码处理异常时,对于检查异常,有两种不同的处理方式:使用try…catch…finally语句块处理它。或者,载寒暑签名中使用throws声明交给函数调用者caller去解决。
try...catch...finally语句块
try{
//...
}catch(SQLException SQLexception){
//...
}catch(Exception exception){
//...
}finally{
。
}
try语句
try块中放可能发生异常的代码。在执行过程中,该段代码可能会产生并抛出一种或几种类型的异常对象,它后面的catch语句要分别对这些异常做相应的处理。如果执行完try且不发生异常,则接着去执行finally块和finally后面的代码(如果有的话)。如果发生异常,则尝试去匹配catch块。
catch语句
每一个catch块用于捕获并处理一个特定的异常,或者这异常类型的子类。Java7中可以将多个异常声明在一个catch中。
catch后面的括号定义了异常类型和异常参数。如果异常与之匹配且是最先匹配到的,则虚拟机将使用这个catch块来处理异常。
在catch块中可以使用这个块的异常参数来获取异常的相关信息。异常参数是这个catch块中的局部变量,其它块不能访问。
如果当前try块中发生的异常在后续的所有catch中都没捕获到,则先去执行finally,然后到这个函数的外部caller中去匹配异常处理器。
如果try中没有发生异常,则所有的catch块将被忽略
finally语句
finally块通常是可选的;无论异常是否发生,异常是否匹配被处理,finally都会执行;一个try至少要有一个catch块,否则,至少要有一个finally块。但是finally不时用来处理异常的,finally不会捕获异常;finally主要做一些清理工作,如流的关闭、数据库连接的关闭等。
throws 函数声明
throws声明:如果一个方法内部的代码会抛出检查异常(checked exception),而方法自己又没有完全处理掉,则javac保证你必须在方法的签名上使用throws关键字声明这些可能抛出的异常,否则编译不通过。
throws是另一种处理异常的方式,它不同于try...catch...finally,throws仅仅是将函数中可能出现的异常向调用者声明,而自己则不具体处理。
采取这种异常处理的原因可能是:方法本身不知道如何处理这样的异常,或者说让调用者处理更好,调用者需要为可能发生的异常负责。
public void foo() throws ExceptionType1 , ExceptionType2 ,ExceptionTypeN
{
//foo内部可以抛出 ExceptionType1 , ExceptionType2 ,ExceptionTypeN 类的异常,或者他们的子类的异常对象。
}
结语
java中,异常处理的任务就是将执行控制流从异常发生的地方转移到能够处理这种异常的地方去。也就是说:当一个函数的某条语句发生异常时,这条语句的后面的语句不会再执行,它失去了焦点。执行流跳转到最近的匹配的异常处理catch代码块去执行,异常被处理完后,执行流会接着在“处理了这个异常的catch代码块”后面接着执行。
为了健壮性的代码,Java异常还需要深入继续学习,加油!