什么是异常
异常指程序运行中出现的不期而至的各种状况,如:文件找不到、网络连接失败、非法参数等。
-
异常的分类
-
检查性异常:
最具代表的检查性异常是用户错误或问题引起的异常,这是程序员无法预见的。例如要打开一个不存在文件时,一个异常就发生了,这些异常在编译时不能被简单地忽略。
-
运行时异常:
运行时异常是可能被程序员避免的异常。与检查性异常相反,运行时异常可以在编译时被忽略。
-
错误ERROR:
错误不是异常,而是脱离程序员控制的问题。错误在代码中通常被忽略。例如,当栈溢出时,一个错误就发生了,它们在编译也检查不到的。
-
异常体系结构
Error和Exception的区别:
- Error通常是灾难性的致命的错误,是程序无法控制和处理的,当出现这些异常时Java虚拟机(JVM)一般会选择终止线程;
- Exception通常情况下是可以被程序处理的,并且在程序中应该尽可能的去处理这些异常。
Java异常处理机制
处理异常的五个关键字:try catch finally throw throws
抛出异常快捷键:Ctrl+Alt+T
-
抛出异常
-
捕获异常
-
举例
public class Test { public static void main(String[] args) { int a=1; int b=0; System.out.println(a/b); }
程序运行后异常名字为:ArithmeticException
使用try-catch-finally处理
try { //监控区域 System.out.println(a/b); }catch(ArithmeticException e){ //捕获异常 System.out.println("程序出现异常,变量b不能为0"); }finally{ // 善后工作 System.out.println("finally"); }
需要处理捕获多个异常时,异常类型从小到大往下写
//假设要捕获多个异常:从小到大 try { //监控区域 System.out.println(a/b); }catch(Error th){ //捕获异常 System.out.println("Error"); }catch (Exception e){ System.out.println("Exception"); } catch (Throwable t){ System.out.println("Throwable"); } finally{ // 善后工作 System.out.println("finally"); } }
主动抛出异常一般使用在方法中,关键字为throw,若该方法无法处理这个异常,则使用throws向上抛出异常
//假设这方法中,处理不了这个异常,方法上抛出异常 public void test(int a,int b) throws ArithmeticException{ if(b==0){//主动抛出异常 throw throws throw new ArithmeticException(); //主动抛出异常,一般在方法中使用 } //System.out.println(a/b); }
自定义异常
使用Java内置的异常类可以描述在编程时出现的大部分异常情况。除此之外,用户还可以自定义异常。用户自定义异常类,只需继承Exception类即可
步骤:
-
创建自定义异常类。
//自定义的异常类 public class MyException extends Exception{ //传递数字>10; private int detail; public MyException(int a) { this.detail=a; } //toString 打印信息 alt+insert @Override public String toString() { return "MyException{" + "detail=" + detail + '}'; } }
-
在方法中通过throw关键字抛出异常对象。
-
在当前抛出异常的方法中处理异常,可以使用try-catch语句捕获并处理;
//可能会存在异常的方法 static void test(int a){ System.out.println("传递的参数为"+a); if(a>10){ try { throw new MyException(a); } catch (MyException e) { System.out.println("MyException=>"+e); } } System.out.println("ok"); } public static void main(String[] args) { test(11); } }
-
否则在方法的声明处通过throws关键字指名要抛出给方法调用者的异常,继续下一步操作,在出现异常的调用者中捕获并处理异常。
public class Test { //可能会存在异常的方法 static void test(int a) throws MyException { System.out.println("传递的参数为"+a); if(a>10){ throw new MyException(a); //抛出 } System.out.println("ok"); } public static void main(String[] args) { try { test(11); } catch (MyException e) { System.out.println("MyException=>"+e); } } }
总结
- 处理运行时异常时,采用逻辑去合理规避同时辅助try-catch处理
- 在多重catch块后面,可以加一个catch(Exception)来处理可能会被遗漏的异常
- 对于不确定的代码,也可以加上try-catch,处理潜在的异常
- 尽量去处理异常,切忌只是简单地调用printStackTrace()去打印输出
- 具体如何处理异常,要根据不同的业务需求和异常类型去决定
- 尽量添加finally语句块去释放占用的资源