-
异常的体系
- 异常机制其实是帮助我们**找到**程序中的问题,异常的根类是`java.lang.Throwable`,其下有两个子类:`java.lang.Error`与`java.lang.Exception`,平常所说的异常指`java.lang.Exception`。
-
异常分类
- 我们平常说的异常就是指Exception,因为这类异常一旦出现,我们就要对代码进行更正,修复程序
- 编译时期异常:checked异常。在编译时期,就会检查,如果没有处理异常,则编译失败。(如日期格式化异常)
- 运行时期异常**:runtime异常。在运行时期,检查异常.在编译时期,运行异常不会编译器检测(不报错)。(如数学异常)
- 编译时期异常和运行时期异常的区别
- 编译时期异常:除了RuntimeException和他的子类,其他都是编译时异常,编译阶段需要进行处理,作用在于提醒程序员
- 运行时期异常:RuntimeException本身和所有子类,都是运行时异常。编译阶段不报错,时程序运行时出现的。一般时由于参数传递错误带来的问题
-
抛出异常throw
- 使用格式
- throw new 异常类名(参数);
throw new NullPointerException("要访问的arr数组不存在");
throw new ArrayIndexOutOfBoundsException("该索引在数组中不存在,已超出范围");
public class ThrowDemo { public static void main(String[] args) { //创建一个数组 int[] arr = {2,4,52,2}; //根据索引找对应的元素 int index = 4; int element = getElement(arr, index); System.out.println(element); System.out.println("over"); } /* * 根据 索引找到数组中对应的元素 */ public static int getElement(int[] arr,int index){ //判断 索引是否越界 if(index<0 || index>arr.length-1){ /* 判断条件如果满足,当执行完throw抛出异常对象后,方法已经无法继续运算。 这时就会结束当前方法的执行,并将异常告知给调用者。这时就需要通过异常来解决。 */ throw new ArrayIndexOutOfBoundsException("哥们,角标越界了```"); } int element = arr[index]; return element; } }
注意:如果产生了问题,我们就会throw将问题描述类即异常进行抛出,也就是将问题返回给该方法的调用者。
那么对于调用者来说,该怎么处理呢?一种是进行捕获处理,另一种就是继
续讲问题声明出去,使用throws声明处理。
- throw new 异常类名(参数);
- 使用格式
-
声明异常throws
-
**声明异常**:将问题标识出来,报告给调用者。如果方法内通过throw抛出了编译时异常,而没有捕获处理(稍后讲解该方式),那么必须通过throws进行声明,让调用者去处理。
关键字**throws**运用于方法声明之上,用于表示当前方法不处理异常,而是提醒该方法的调用者来处理异常(抛出异常).
-
格式:修饰符 返回值类型 方法名(参数) throws 异常类名1,异常类名2…{ }
-
代码演示:
public class ThrowsDemo { public static void main(String[] args) throws FileNotFoundException { read("a.txt"); } // 如果定义功能时有问题发生需要报告给调用者。可以通过在方法上使用throws关键字进行声明 public static void read(String path) throws FileNotFoundException { if (!path.equals("a.txt")) {//如果不是 a.txt这个文件 // 我假设 如果不是 a.txt 认为 该文件不存在 是一个错误 也就是异常 throw throw new FileNotFoundException("文件不存在"); } } } //throws用于进行异常类的声明,若该方法可能有多种异常情况产生,那么在throws后面可以写多个异常类,用逗号隔开。 public class ThrowsDemo2 { public static void main(String[] args) throws IOException { read("a.txt"); } public static void read(String path)throws FileNotFoundException, IOException { if (!path.equals("a.txt")) {//如果不是 a.txt这个文件 // 我假设 如果不是 a.txt 认为 该文件不存在 是一个错误 也就是异常 throw throw new FileNotFoundException("文件不存在"); } if (!path.equals("b.txt")) { throw new IOException(); } } }
-
-
捕获异常try...catch
-
异常出现会立刻终止程序。所以我们要处理异常
-
在方法中使用try-catch的方式就是捕获异常
-
捕获异常:Java中对异常有针对性的语句进行捕获,可以对出现的异常进行指定方式的处理。
-
语法:try{
编写可能会出现异常的代码
}catch(异常类型 e){
处理异常的代码
//记录日志/打印异常信息/继续抛出异常
} - 演示:
public class TryCatchDemo { public static void main(String[] args) { try {// 当产生异常时,必须有处理方式。要么捕获,要么声明。 read("b.txt"); } catch (FileNotFoundException e) {// 括号中需要定义什么呢? //try中抛出的是什么异常,在括号中就定义什么异常类型 System.out.println(e); } System.out.println("over"); } /* * * 我们 当前的这个方法中 有异常 有编译期异常 */ public static void read(String path) throws FileNotFoundException { if (!path.equals("a.txt")) {//如果不是 a.txt这个文件 // 我假设 如果不是 a.txt 认为 该文件不存在 是一个错误 也就是异常 throw throw new FileNotFoundException("文件不存在"); } } }
-
-
关于异常处理的四个问题
-
如果try中没有遇到问题,如何执行?
会把try里面所有的代码全部执行完毕,不会执行catch里面的代码
如果try中可能会遇到索格问题,怎么执行?
写多个catch与之对应,父类异常需要写在下面
如果try中遇到的问题没有被捕获,怎么执行?
相当于try...catch白写了,当前异常会交给虚拟机处理
如果try中可能会遇到多个问题,那么try下面其他代码还会执行吗?
不会执行了,try中遇到问题,直接跳转到对应的catch,如果没有对应的catch与之匹配,则交给虚拟机。
-
-
自定义异常
- 我们为什么要自定义异常?
- 答:我们说了Java中不同的异常类,分别表示着某一种具体的异常情况,那么在开发中总是有些异常情况是SUN没有定义好的,此时我们根据自己业务的异常情况来定义异常类。
- 演示
// 业务逻辑异常 public class LoginException extends Exception { /** * 空参构造 */ public LoginException() { } /** * * @param message 表示异常提示 */ public LoginException(String message) { super(message); } } public class Demo { // 模拟数据库中已存在账号 private static String[] names = {"bill","hill","jill"}; public static void main(String[] args) { //调用方法 try{ // 可能出现异常的代码 checkUsername("nill"); System.out.println("注册成功");//如果没有异常就是注册成功 } catch(LoginException e) { //处理异常 e.printStackTrace(); } } //判断当前注册账号是否存在 //因为是编译期异常,又想调用者去处理 所以声明该异常 public static boolean checkUsername(String uname) throws LoginException { for (String name : names) { if(name.equals(uname)){//如果名字在这里面 就抛出登陆异常 throw new LoginException("亲"+name+"已经被注册了!"); } } return true; } }
- 我们为什么要自定义异常?