文章目录
一、异常概述与异常体系结构
1.定义
在Java语言中,将程序执行中发生的不正常情况称为“异常”。 (开发过程中的语法错误和逻辑错误不是异常)
2.分类
(1)
Error:Java虚拟机无法解决的严重问题。如:JVM系统内部错误、资源耗尽等严重情况。比如:StackOverflowError(栈溢出)和OOM(OutOfMemoryError,堆溢出)。一般不编写针对性的代码进行处理。
(2)
Exception: 其它因编程错误或偶然的外在因素导致的一般性问题,可以使用针对性的代码进行处理。例如:
- 空指针访问
- 试图读取不存在的文件
- 网络连接中断
- 数组角标越界
3.异常体系结构
捕获错误最理想的是在编译期间,但有的错误只有在运行时才会发生。比如:除数为0,数组下标越界等 。
Exception异常又分为:编译时异常(受检(checked)异常)和运行时异常(非受检(unchecked)异常)。
java.lang.RuntimeException类及它的子类都是运行时异常。
二、异常处理机制
一.try-catch-finally
1.理解
Java提供的是异常处理的抓抛模型。
“抛” :程序在正常执行的过程中,一旦出现异常,就会在异常代码处生成一个对应异常类的对象。并将此对象抛出,该异常对象将被提交给Java运行时系统。一旦对象抛出后,其后的代码就不再执行。
“抓” :可以理解为异常的处理方式。
2.语句格式
try
{
//可能出现异常的代码
}
catch(异常类型 变量名)
{
//异常处理
}
finally
{
//一定会执行的代码
}
(1)
可以写多个 catch ;
(2)
finally 是可选的(同C#);
<1>
不论在try代码块中是否发生了异常事件,catch语句是否执行,catch语句是否有异常,catch语句中是否有return,finally块中的语句都会被执行。 如图,若采用①,则运行时不会输出,而采用②,会输出“我好帅啊~~”
<2>
像数据库连接、输入输出流、网络编程Socket等资源,JVM是不能自动回收的,需要自己手动进行资源释放。此时的资源释放就需要声明在finally中。
(3)
使用try将可能出现异常的代码包装起来,在执行过程中,一旦出现异常,就会生成一个对应异常类的对象,根据此对象的类型,去catch中进行匹配。
(4)
一旦try中的异常对象匹配到某一个catch时,就进入catch中进行异常的处理。一旦处理完成,就跳出当前的try-catch结构(此时没有finally),继续执行其后的代码。
(5)
catch中的异常类型如果满足子父类关系,则要求子类一定声明在父类的上面,否则报错。
(6)
常用的异常对象处理方式: ① String getMessage()获取异常信息,返回字符串 ② printStackTrace()获取异常类名和异常信息,以及异常出现在程序中的位置。返回值void。
(7)
在try结构中声明的变量,出了try结构后不能再被调用。
(8)
try-catch-finally 可以嵌套。
使用try-catch-finally处理编译时异常,使得程序在编译时不再报错,但是运行时仍可能报错。相当于将一个编译时可能出现的异常延迟到运行时出现。
二.throws + 异常类型
1.理解
“throws + 异常类型” 写在方法的声明处,指明此方法执行时,可能会抛出异常类型。一旦当方法体执行时,出现异常,仍会在异常代码处生成一个异常类的对象,此对象满足throws后异常类型时,就会被抛出。异常代码后续的代码就不再执行。
try-catch-finally 真正将异常处理掉了,但是throws 的方法只是将异常抛给了方法的调用者,并没有真正将异常处理掉。
2.格式
public void readFile(String file) throws FileNotFoundException {
……
// 读文件的操作可能产生FileNotFoundException类型的异常
FileInputStream fis = new FileInputStream(file);
..……
}
3.重写方法声明抛出异常的原则
子类重写方法不能抛出比被父类被重写的方法范围更大的异常类型。
如:
public class A {
public void methodA() throws IOException {
……
} }
public class B1 extends A {
public void methodA() throws FileNotFoundException {
……
} }
public class B2 extends A {
public void methodA() throws Exception { //报错
……
} }
这就是说,父类中被重写的方法没有throws方式处理异常,则子类重写的方法也不能使用throws。意味着若子类重写的方法中有异常,必须使用try-catch-finally 方式处理。
三.开发中如何选择这两种处理机制
(1)
父类中被重写的方法没有throws方式处理异常,则子类重写的方法也不能使用throws。意味着若子类重写的方法中有异常,必须使用try-catch-finally 方式处理。
(2)
执行的方法a 中,先后又调用了另外的几个方法,这几个方法是递进关系执行的。我们建议这几个方法使用throws的方式进行处理。而执行的方法a可以考虑使用try-catch-finally 方式处理。
三、手动抛出异常
1.理解
Java异常类对象除在程序执行过程中出现异常时由系统自动生成并抛出,也可根据需要使用人工创建并抛出。
2.格式
首先要生成异常类对象,然后通过throw语句实现抛出操作(提交给Java运行环境)。
IOException e = new IOException();
throw e;
可以抛出的异常必须是Throwable或其子类的实例。下面的语句在编译时将会产生语法错误:
throw new String("want to throw");
String 不是异常类。
四、用户自定义异常类
1.理解
用户自定义异常类MyException,用于描述数据取值范围错误信息。用户自己的异常类必须继承现有的异常类。
class MyException extends Exception {
static final long serialVersionUID = 13465653435L;
public MyException() {
super();
public MyException(String message) {
super(message);
}
2.如何自定义异常类
<1>
继承于现有的异常结构:RuntimeException、 Exception;
<2>
提供全局常量:serialVersionUID
<3>
提供重载的构造器