异常概述
- 异常:程序执行中发生的不正常情况。
- 开发过程中的语法错误不叫异常
- Java的异常处理:处理非预期的情况,如文件没找到,空指针等。
- Java程序运行过程中所发生的异常事件可分为两类:
- Error: JVM系统内部错误、资源耗尽等严重情况 (无法控制,不处理)
- Exception:其它因编程错误或偶然的外在因素导致的一般性问题。比如:空指针访问、试图读取不存在的文件等。是程序本身可以处理的异常
异常分类
- Java的异常(包括Exception和Error)分为可查的异常(checked exceptions)和不可查的异常(unchecked exceptions)
- 可查的异常(编译器要求必须处置的异常)
- 不可查的异常(编译器不要求强制处置的异常)
- Exception 这种异常分两大类:运行时异常和非运行时异常(编译异常)
- 运行时异常:都是 RuntimeException 类及其子类异常
- 非运行时异常(编译异常):是 RuntimeException 以外的异常
常见异常
- RuntimeException
- 错误的类型转换
- 数组下标越界
- 算数异常
- 空指针访问
- IOExeption
- 从一个不存在的文件中读取数据
- 越过文件结尾继续读取
- 连接一个不存在的URL
异常处理
异常处理机制
异常处理是通过try-catch-finally语句实现的。
try{
… //可能产生异常的代码
}catch(ExceptionName1e){
… //当产生ExceptionName1型异常时的处置措施
}catch(ExceptionName2e){
… //当产生ExceptionName2型异常时的处置措施
} [finally {
… //无条件执行的语句
}]
- try:将可能出现异常的代码放在try语句块中。
- catch (Exceptiontype e):在catch语句块中是对异常对象进行处理的代码。getMessage(); printStrackTrace():打印异常信息(程序出错的位置信息及原因)
- catch代码块可以有多个,try中抛出的异常对象是从上往下去匹配catch中的异常类型,一旦满足就执行catch中的代码。执行完毕,就跳出所有的catch。
- 如果catch中多个异常类型是并列关系,不需要考虑顺序问题。如果catch中多个异常类型有继承关系,子类需要在父类异常对象的上面。
- 如果明确知道产生的是何种异常,可以用该异常类作为catch的参数也可以用其父类作为catch的参数。
- 每个try语句块可以伴随一个或多个catch语句,用于处理可能产生的不同类型的异常对象。
异常处理语法规则
- 必须在try 之后添加catch 或finally 块。
- 必须遵循块顺序:若代码同时使用catch 和finally 块,则必须将catch 块放在try 块之后。
- catch 块与相应的异常类的类型相关。
- 一个try 块可能有多个catch 块。
- 可嵌套try-catch-finally 结构。
- 在try-catch-finally 结构中,可重新抛出异常。
异常处理一般原则
- 能处理就早处理,抛不出去还不能处理的就想法消化掉或者转换为RuntimeException处理。
- 事务异常,在持久化层、Dao层只能往上抛。
- 对于检查异常,如果不能行之有效的处理,还不如转换为RuntimeException抛出。这样也让上层的代码有选择的余地――可处理也可不处理。
- 对于一个应用系统来说,应该有自己的一套异常处理框架,这样当异常发生时,也能得到统一的处理风格,将优雅的异常信息反馈给用户。
try、catch、finally语句块的执行
抛出异常
throws语句的语法格式为:
methodname throws Exception1,Exception2,..,ExceptionN { }
Throws抛出异常的规则:
- 如果是不可查异常(unchecked exception),即Error、RuntimeException或它们的子类,那么可以不使用throws关键字来声明要抛出的异常,编译仍能顺利通过,但在运行时会被系统抛出。
- 必须声明方法可抛出的任何可查异常(checked exception)。
- 仅当抛出了异常,该方法的调用者才必须处理或者重新抛出该异常。
- 调用方法必须遵循任何可查异常的处理和声明规则。
Throw抛出异常:
- throw总是出现在函数体中,用来抛出一个Throwable类型的异常
- 语法格式为:throw new exceptionname;