异常类
程序中出现的错误分为两类:语法错误和语义错误
语法错误能够被及时发现,并做出错误提示,因此又称为编译器错误
语义错误往往难以被发现,甚至错误的出现具有随机性和不确定性,因此又叫运行期错误
运行期错误又可以分为两类:错误(Error)和异常(Exception)
前者是虚拟机本身或者系统本身出现的错误,例如递归调用层数太深而导致堆栈溢出错误,系统内存不足而导致的内存溢出错误等。
后者往往是我们程序自身出现的错误,例如空指针异常错误(NullPointerException)、数组越界错误(ArrayIndexOutOfBoundsException)、算术运算错误等
注意: 上面例子里的数组是类类型数组,数组中的元素都是类A的引用型变量,没有用new运算符分配变量的话不能使用,否则会出现空指针异常错误。
所谓异常就是程序运行时可能出现一些错误,比如试图打开一个根本不存在的文件等,异常处理将会改变程序的控制流程,让程序有机会对错误作出处理。
异常对象可以调用如下方法得到或输出有关异常的信息:
public String getMessage();
public void printStackTrace();
public String toString();
处理异常
try~catch语句
将可能出现异常的代码放在try~catch语句的try部分,当try中代码在运行期间实际发生异常后,try部分将立刻结束执行,程序跳到catch语句里面执行。
try~catch语句可以由几个catch组成,分别处理相应的异常。
catch的作用不一定在于修复错误,而是输出错误信息,执行完catch语句后,不会回到try语句中执行剩下的语句,程序会接着catch后面往下执行(不终止)。
多个catch语句,当try语句块中发生异常,catch异常捕获自上而下进行匹配,只执行第一个匹配的catch语句,所以catch参数中的异常类应该按照 继承关系 ,将子类异常放在前面,父类异常放在后面。
格式
try{
...
}
catch(Exception e){
...
}
try{
...
}
catch(NumberFormatException e){
...
}
catch(NullPointerException e){
...
}
catch(Exception e){ //前面几个异常类都是Exception的子类
...
}
try~catch语句的嵌套
1.当try中出现异常,后面的多个catch都不匹配的时候,会向上层catch中寻找匹配
2.若没有上层,或所有上层都不匹配,程序将停止运行
finally子语句
带finally子语句的try~catch语句,格式
try{}
catch(Exception e){}
finally{}
其执行机制是:在执行try~catch语句后,执行finally子语句,也就是说,无论try部分是否发生异常,finally子语句都会被执行。
注意两个特殊情况:
1.如果try~catch语句中执行了return语句后,finally语句仍然被执行
2.try~catch语句中执行了程序退出代码后,即执行System.exit(0);则不执行finally子语句
自己抛出异常
1.若明确某方法执行时有抛出异常的可能,需要在方法声明处加throws关键词,其含义是告诉方法的调用者,该方法可能会抛出的异常类型。并在该方法体中具体给出产生异常的操作,即用相应的异常类创建对象,并用throw关键字抛出异常对象,导致该方法结束执行。
2.throws关键词后声明异常类型可以多个,用逗号隔开
3.throws声明中的异常类可以是实际throw(抛出)的类的父类
4.程序在try~catch语句的try块中调用能发生异常的方法,catch捕获throw抛出的异常
5.注意区分throw和throws,throw的作用是抛出异常(对象),throws是声明方法可能抛出的异常(对象)的类型
自定义异常类
我们也可以扩展Exception类定义自己的异常类,然后规定哪些方法产生这样的异常类的异常(对象)。
class A{int x;}
class MyException extends Exception{
public String getMessage(){
System.out.println(“出现了自定义异常”);
}
}
class B{
public static void f1() throws MyException{
throw new MyException();
}
public static void main(String[] args){
f1();
}
}