异常的基本语法:
try{
有可能出现异常的语句 ;
}[catch (异常类型 异常对象) {
e.printStackTrace();//打印出异常的调用栈
} ... ]
[finally {
异常的出口
}]
- try代码块中放的是可能出现异常的代码.try的()中可以创建Scanner,可以负责回收资源,保证try结束时自动调用Scanner的close方法.(alt+enter 优化代码)
- catch 代码块中放的是出现异常后的处理行为.
- finally 代码块中的代码用于处理善后工作, 会在最后执行.但是不建议在finally中写return语句,
- 其中 catch 和 finally 都可以根据情况选择加或者不加.
- catch可以有多个
- 一个catch捕获所有异常catch (Exception e),这里的Exception就是所有异常的父类,所以具备这样的功能
- finally表示善后工作,无论是否有异常,finally都是会被执行到的
异常处理流程
- 程序先执行 try 中的代码
- 如果 try 中的代码出现异常, 就会结束 try 中的代码, 看和 catch 中的异常类型是否匹配.
- 如果找到匹配的异常类型, 就会执行 catch 中的代码
- 如果没有找到匹配的异常类型, 就会将异常向上传递到上层调用者.
- 无论是否找到匹配的异常类型, finally 中的代码都会被执行到(在该方法结束之前执行).
- 如果上层调用者也没有处理的了异常, 就继续向上传递.
- 一直到 main 方法也没有合适的代码处理异常, 就会交给 JVM 来进行处理, 此时程序就会异常终止.
抛出异常
使用throw
throw new ArithmeticException("抛出除 0 异常");
使用throws(标注在方法定义的位置上)
public static int divide(int x, int y) throws ArithmeticException {}
java异常体系
如果一段代码可能抛出 受查异常, 那么必须显式进行处理.
自定义异常
- 可以基于已有的异常类进行扩展(继承), 创建和
我们业务相关的异常类. - 自定义异常通常会继承自 Exception 或者 RuntimeException
- 继承自 Exception 的异常默认是受查异常
- 继承自 RuntimeException 的异常默认是非受查异常.
public class Test0 {
private static String userName = "haha";
private static String password = "666666";
public static void main(String[] args) {
try {
login("haha", "666666");
} catch (UserError userError) {
userError.printStackTrace();
} catch (PasswordError passwordError) {
passwordError.printStackTrace();
}
}
public static void login(String userName,String password)
throws UserError,PasswordError{
if(!Test0.userName.equals(userName)){
throw new UserError("用户名错误");
}
if(!Test0.password.equals(password)){
throw new PasswordError("密码错误");
}
System.out.println("登录成功!");
}
}
//创建相关异常类
class UserError extends Exception{
public UserError(String message){
super(message);
}
}
class PasswordError extends Exception{
public PasswordError(String message){
super(message);
}
}