在程序中,异常可能由程序员没有预料到的各种情况产生,也可能超出了程序员可控范围的环境因素产生,如用户的坏数据、试图打开一个根本不存在的文件等。在Java中,这种在程序运行时可能出现的一些错误称为异常。异常是一个在程序执行期间发生的事件,它中断了正在执行的程序的正常指令流。
【例9.1】0可以作为除数么?
代码及运行结果如下:
程序运行的结果报告发生了算术异常ArithmeitcException(根据给出的错误提示可知,发生错误是因为在算术表达式“3/0”中,0作为除数出现),系统不再执行下去,提前结束。这种情况就是所说的异常。
有许多异常的例子,如数组溢出等。Java语言是一门面向对象的编程语言,因此异常在Java语言中也是作为类的实例的形式出现的。当某一方法中发生错误时,这个方法会创建一个对象,并且把它传递给正在运行的系统。这个对象就是异常对象。通过异常处理机制,可以将非正常情况下的处理代码与程序的主逻辑分离,即在编写代码主流程的同时在其他地方处理异常。
9.2异常的抛出与捕捉
为了保证程序有效地执行,需要对抛出异常进行相应的处理。在Java中,如果某个方法抛出异常,既可以在当前方法中进行捕捉,而后处理该异常,也可以将异常向上抛出,交由方法调用者来处理,下面讲解Java中捕获异常的方法。
9.2.1抛出异常
异常抛出后,如果不做任何处理,程序就会被终止。例如,将一个字符串转换为整型,可以通过Integer类的parseInt()方法来实现。但如果该字符串不是数字形式,parsrInt()方法就会抛出异常,程序将在异常的位置终止,不再执行下面的语句。
【9.2】控制台输出“lili年龄是:20L”
在项目中创建Thundering类,在主方法中实现将不是数字形式的字符串转换为int型。运行程序,系统会报出异常提示。实例代码如下:
运行代码截图:
1.try-catch语句块
下面将例9.2中的代码进行修改
【例9.3】捕获例9.2中主方法抛出的异常
在项目中创建Take类,在主方法中使用try-catch语句块将可能出现的异常语句进行异常处理,实例代码如下:
运行结果截图:
从截图看出,程序仍然输出最后的提示信息,没有因为异常而终止,程序就会跳转到catch语句块中执行,执行完catch语句块中的程序代码后,将继续执行catch 语句块后的其他代码,而不会执行try语句块中发生异常语句后面的代码。由此可知,Java的异常处理是结构化处理的不会因为一个异常影响整个程序的执行。
误区警示:
有时为了编程简单会忽略catch语句后的代码,这样try-catch语句就成了一种摆设,一旦程序在运行过程中出现了异常,就会导致最终运行结果与期望的不一致,而错误发生的原因很难查找。因此要养成良好的编程习惯,最好的catch语句块中写入处理异常的代码。
2.finally语句块
完整的异常处理语句一定要包含finally语句,无论程序中有无异常发生,并且无论之前的try-catch语句块是否顺理执行完毕,都会执行finally语句。但是,在以下四种特殊情况下finally不会被执行:
1.在finally语句块中发生了异常
2.在前面的代码中使用了System.exit()退出程序。
3.程序所在的线程死亡
4.关闭CPU
9.3Java常见的异常类
在Java中,提供了一些异常类用来描述经常发生的异常。其中有的需要程序员进行捕捉处理或声明抛出,有的是Java虚拟机自动进行捕获处理。Java中常见的异常类如表:
异常类 说明
ClassCastException 类型转换异常
ClassNotFoundException 未找到相应类异常
ArithmeticException 算数异常
ArrayIndexOutOfException 数组下标越界异常
SQLException 操作数据库异常类
NullPointerException 空指针异常
NoSuchFieldException 字段未找到异常
NumberFormatException 方法未找到抛出的异常
NumberFormatExcesption 字符串转换为数字抛出的异常
NegativeArraySizeException 数组元素个数为负数抛出的异常
StringIndexOutOfBoundException 字符串索引超出范围抛出的异常
IOException 输入输出异常
IllegalAccessException 不允许访问某类异常
InstantiationException 当应用程序试图使用Class类中的newInstance()方法创建一个类的实例,而指定的类对象无法被实例化时,抛出该异常
EOFException 文件已结束异常
Exception 文件未找到异常
ArrayStoreException 数组中包含不兼容的值抛出的异常
9.4自定义异常
使用Java内置的异常类可以描述在编程时出现的大部分异常情况。除此之外,用户只需要继承Exception类即可自定定义异常类。在程序中使用自定义异常类,大体可分为以下几个步骤:
(1)创建自定义异常类
(2)在方法中通过throw关键字抛出异常对象
(3)如果在当前抛出异常的方法中处理异常,可以使用try-catch语句块捕获并处理,否则在方法的声明出通过throws关键字指明要抛出给方法调用者的异常,继续进行下一步操作。
(4)再出现异常的方法的调用者中捕获并处理异常
【例9.4】如何让创建自定义异常
在项目中创建MyException类,该类继承Exception类。实例代码如下:
package Nine_exercises;
public class MyException extends Exception{//创建自定义异常,继承Exception
public MyException(String ErrorMessage) {//构造方法
super(ErrorMessage);//父类构造方法
}
}
字符串ErrorMessage是要输出的错误信息。若想抛出用户自定义的异常对象,要使用thorw关键字【例9.5】自定义异常的抛出与捕捉
在项目中创建Tran类,在该类中创建一个带有int型参数的方法avg(),该方法用来检查是否小于0或大于100.如果参数小于0或大于100,则通过throw关键字抛出一个MyException异常对象,并在main()方法中捕捉该异常。实例代码如下:
package Nine_exercises;
public class Tran {//创建类
static int avg(int number1,int number2)throws MyException{//定义方法抛出异常
if(number1<0||number2<0){判断方法中参数是否满足指定条件
throw new MyException("不可以使用负数");//抛出错误信息
}
if(number1>100||number2>100) {//判断方法中参数是否满足指定条件
throw new MyException("数值太大了");//抛出错误信息
}
return(number1+number2)/2;//将参数的平均值返回
}
public static void main(String[] args) {//主方法
try {//try语句块处理可能出现异常的代码
int result=avg(102,150);//调用avg()方法
System.out.println(result);//将avg()方法的返回值输出
}catch(MyException e) {
System.out.println(e);//输出异常信息
}finally {
System.out.println("程序结束");//提示结束
}
}
}
9.5在方法中抛出异常
若某个方法可能会发生异常,但不想再当前方法中处理这个异常,则可以使用throws、thorw关键字在方法中抛出异常。
9.5.1使用throws关键字抛出异常
throws关键字通常被应用在声明方法时,用来指定方法可能抛出的异常。多个异常可使用逗号分号隔开。
【例9.6】指明异常起源于何处
在项目中创建Shoot类,该类中创建方法pop(),在该方法中抛出NwgativeArraySizeException异常,在主方法中调用该方法并实现异常处理。示例代码如下:
package Nine_exercises;
public class Shoot {//创建类
static void pop()throws NegativeArraySizeException{
//定义方法并抛出NegativeArraySizeException
int[]arr=new int[-3];//创建数组
}
public static void main(String[] args) {//try语句处理异常信息
try {//调用pop()方法
pop();
}catch(NegativeArraySizeException e) {
System.out.println("pop()方法抛出的异常");//输出异常信息
}
}
}
运行截图如下:
使用throws关键字将异常抛给上一级,如果不想处理该异常,可以继续向上抛出,但最终要有能够处理该异常的代码。
说明:如果是Error类、RuntimeException类或它们的子类,可以不使用throw关键字来声明要抛出的异常,编译仍能顺利通过,但在运行时会被系统抛出。
9.5.2使用throw关键字抛出异常
throw关键字通常用于方法体中,并且抛出一个异常对象。程序在执行到throw语句时立即终止,他后面的 语句都不执行。通过throw抛出异常后,如果想在上以及代码中捕获并处理异常,则需要在抛出异常的方法中使用throws关键字在方法的声明中指明要抛出的异常;如果要捕捉throw抛出的异常,则必须使用try-catch语句块。
【例9.7】创建自定义异常
在项目中创建自定义异常类(MyException类),继承Exception类。实例代码如下:
package Nine_exercises;
public class MyException extends Exception{//创建自定义异常,继承Exception
String message;//定义String
public MyException(String ErrorMessage) {//父类方法
message=ErrorMessage;//父类方法
}
public String getMessage() {//覆盖getMessage方法
return message;
}
}
【例9.8】使用throw关键字捕捉自定义异常
使用throw关键字捕捉异常。在项目中创建Captor类,该类中的quotient()方法传递两个int型参数,如果其中的一个参数为负数,则会抛出MyException异常,最后在mian()方法中捕捉异常。实例代码如下:
package Nine_exercises;
public class Captor {
static int quotient(int x,int y)throws MyException{//定义方法抛出异常
if(y<0) {//判断参数是否小于0
throw new MyException("除数不能是负数");//异常信息
}
return x/y;//返回值
}
public static void main(String[] args) {//主方法
try {//try语句块包含可能发生的异常的语句
int result=quotient(3,-1);//调用方法quotient
System.out.println(result);//输出result的值
}catch(MyException e) {//处理自定义异常
System.out.println(e.getMessage());//输出异常信息
}catch(ArithmeticException e) {//处理ArithmeticException
System.out.println("除数不能为0");//输出提示信息
}catch(Exception e) {//处理其他异常
System.out.println("程序发生了其他异常");//输出提示信息
}
}
}
截图:
9.6运行时异常
Runtime Exception异常是程序运行过程中抛出的异常。Java类库的每个包中定义了异常类,所有这些类都是Throwable类的子类。Throwable类派生了两个子类,分别是Exception类和Error类。Error类及其子类用来描述Java运行系统中的内部错误以及资源耗尽的错误,这类错误比较严重。Exception类称为非致命性类,可以通过捕捉处理使程序继续执行。Exception类又根据错误发生的愿意分为RuntimeException异常和除RuntimeException之外的异常,如图9所示。
Java中提供了常见的RuntimeException异常,这些异常可通过try-catch语句捕获,如下表:
种类 异常
NullPointerException 空指针异常
ArrayIndexOutBoundsException 数组下标越界
ArithmetivException 算数异常
ArrayStoreException 数组中包含不兼容的值抛出的异常
IllegalArgumentException 非法参数异常
SecurityException 安全性异常
NegativeArraySizeException
数组长度为负异常
9.7异常的使用原则
Java异常强制用户去考虑程序的强健性和安全性 。异常处理不应用来控制程序的正常流程,其主要作用是捕获程序在运行时发生的异常并进行相应的处理。编写代码处理某个方法可能出现的异常时,可遵循以下几条原则:
1.在当前方法中声明使用try-catch语句捕获异常。
2.一个方法被覆盖是,覆盖它的方法必须抛出相同的异常或异常的子类。
3.如果父类抛出多个异常,则覆盖方法必须抛出那些异常的一个子集,不能跑出新异常