异常的基本用法
捕获异常
基本语法
try{
有可能出现异常的语句 ;
}[catch (异常类型 异常对象) { } ... ]
[finally {
异常的出口
}]
- try 代码块中放的是可能出现异常的代码.
- catch 代码块中放的是出现异常后的处理行为.
- finally代码块中的代码用于处理善后工作, 会在最后执行.
- catch 和 finally 都可以根据情况选择加或者不加
- 方法之间是存在相互调用关系的, 这种调用关系我们可以用 “调用栈” 来描述. 在 JVM 中有一块内存空间称为 “虚 拟机栈” 专门存储方法之间的调用关系. 当代码中出现异常的时候, 我们就可以使用 e.printStackTrace(); 的 方式查看出现异常代码的调用栈.
- 一段代码可能会抛出多种不同的异常, 不同的异常有不同的处理方式. 因此可以搭配多个 catch 代码块. 如果多个异常的处理方式是完全相同, 也可以写成这样
- Exception 类是所有异常类的父类.可以用这个类型表示捕捉所有异常.
- finally 表示最后的善后工作, 例如释放资源,无论是否存在异常, finally 中的代码一定都会执行到. 保证最终一定会执行到 Scanner 的 close 方法.
异常处理流程
- 程序先执行 try 中的代码
- 如果 try 中的代码出现异常, 就会结束 try 中的代码, 看和 catch 中的异常类型是否匹配.
- 如果找到匹配的异常类型, 就会执行 catch 中的代码
- 如果没有找到匹配的异常类型, 就会将异常向上传递到上层调用者.
- 无论是否找到匹配的异常类型, finally 中的代码都会被执行到(在该方法结束之前执行).
- 如果上层调用者也没有处理的了异常,就继续向上传递. 一直到 main 方法也没有合适的代码处理异常, 就会交给 JVM 来进行处理, 此时程序就会异常终止.
异常说明
我们在处理异常的时候, 通常希望知道这段代码中究竟会出现哪些可能的异常.
我们可以使用 throws 关键字, 把可能抛出的异常显式的标注在方法定义的位置. 从而提醒调用者要注意捕获这些异常.
Java 异常体系
Java 内置了丰富的异常体系, 用来表示不同情况下的异常.
- 顶层类 Throwable 派生出两个重要的子类, Error 和 Exception
- 其中 Error 指的是 Java运行时内部错误和资源耗尽错误. 应用程序不抛出此类异常. 这种内部错误一旦出现, 除了告知用户并使程序终止之外, 再无能无力. 这种情况很少出现.
- Exception 是我们程序猿所使用的异常类的父类.
- 其中 Exception 有一个子类称为RuntimeException , 这里面又派生出很多我们常见的异常类 NullPointerException , IndexOutOfBoundsException 等.
Java语言规范将派生于 Error 类或 RuntimeException 类的所有异常称为 非受查异常, 所有的其他异常称为 受查异常.
自定义异常类
- 自定义异常通常会继承自 Exception 或者 RuntimeException
- 继承自 Exception 的异常默认是受查异常
- 继承自 RuntimeException 的异常默认是非受查异常.
实现一个用户登陆功能.
ublic class Users {
private static String userName = "admin";
private static String password = "admin";
public static void main(String[] args) {
try {
login();
} catch (UsererrorException userError) {
userError.printStackTrace();
} catch (PassworderrorException passwordError) {
passwordError.printStackTrace();
}
}
public static void login() throws PassworderrorException,
UsererrorException {
Scanner scanner = new Scanner(System.in);
while ( true ) {
System.out.println("请输入用户名:");
String userName = scanner.nextLine();
System.out.println("请输入密码:");
String password = scanner.nextLine();
if ( ! users.userName.equals(userName) ) {
throw new UsererrorException("用户名错误!");
}
if ( ! users.password.equals(password) ) {
throw new PassworderrorException("密码错误!");
}
System.out.println("登陆成功");
}
}
}
class UsererrorException extends Exception {
public UsererrorException(String message) {
super(message);
}
}
class PassworderrorException extends Exception {
public PassworderrorException(String message) {
super(message);
}
}