Java异常机制

132 阅读4分钟

什么是异常

  • 实际工作中,遇到的情况不可能是非常完美的。比如:你写的某个模块,用户输入不一定符合你的要求;你的程序要打开某个文件,这个文件可能不存在或者文件格式不对;你要读取数据库的数据,数据可能是空的等。我们的程序再跑着,内存或者硬盘可能满了。等等。
  • 软件程序在运行过程中,非常可能遇到刚刚提到的这些异常问题,我们叫异常,Exception,怎么让我们写的程序做出合理的处理,而不至于程序崩溃。
  • 异常指程序运行中出现的不期而至的各种情况:比如文件找不到,网络连接失败,非法参数等。
  • 异常发生在程序运行期间。它影响了正常的程序执行流程。

简单分类

要理解Java异常是如何工作的,你需要掌握以下三种类型的异常:

  1. 检查性异常:最具代表性的检查性异常是用户错误或问题引起的异常,这是程序员无法预见的。例如要打开一个不存在文件时,一个异常就发生了,这些异常在编译时不能被简单地忽略。
  2. 运行时异常:运行时异常是可能被程序员避免的异常。与检查性异常相反,运行时异常可以在编译时被忽略。
  3. 错误:错误不是异常,而是脱离程序员控制的问题。错误在代码中通常被忽略。例如,当栈溢出时,一个错误就发生了,它们在编译也检查不到的。

异常体系架构

Java把异常当作对象来处理,并定义一个基类java.lang.Throwable作为所有异常的超类。

在Java API中已经定义了许多异常类,这些异常类分为两大类,错误Error和异常Exception。

Error和Exception的区别:Error通常是灾难性的致命的错误,是程序无法控制和处理的,当出现这些异常时,JVM一般会选择终止线程;Exception通常情况下是可以被程序处理的,并且在程序中应该尽可能的去处理这些异常。

Java异常处理机制

  • 抛出异常
  • 捕获异常
  
  package com.oop.exception;
  ​
  public class Test {
      public static void main(String[] args) {
          int a = 1;
          int b = 0;
          //try和catch必须要有 finally可以没有
          try {//try监控区域
              System.out.println(a / b);
          } catch (ArithmeticException e) {//catch 捕获异常
              System.out.println("程序出现异常 b不能为0");
          } finally {//处理善后工作 一定会被执行
              System.out.println("finally");
          }
      } 
  }

假设要捕获多个异常:从小到大。即大的异常要放在后面;

  
  package com.oop.exception;
  ​
  public class Test {
      public static void main(String[] args) {
          int a = 1;
          int b = 0;
          //try和catch必须要有 finally可以没有
          try {//try监控区域
              System.out.println(a/b);;
          } catch (Error e) {//catch 捕获异常  把最大的异常写在最后面
              System.out.println("Error");
          }catch (Exception e) {//catch 捕获异常
              System.out.println("Exception");
          }catch (Throwable e) {//catch 捕获异常
              System.out.println("Throwable");
          } finally {//处理善后工作 一定会被执行
              System.out.println("finally");
          }
      }
  }
  /*结果:
  Exception
  finally
  */

快捷键生成try catch finall代码:选中语句Ctrl+Alt+T

  
  package com.oop.exception;
  ​
  public class Test2 {
      public static void main(String[] args) {
          int a = 1;
          int b = 0;
          //Ctrl+Alt+T
          try {
              System.out.println(a/b);
          } catch (Exception e) {
              System.exit(1);//结束程序
              e.printStackTrace();//打印错误的栈信息
          } finally {
          }
      }
  }

throw与throws:

  
  package com.oop.exception;
  ​
  public class Test2 {
      public static void main(String[] args) {
         new Test2().test(1,0);
      }
      public void test(int a,int b){
          if (b==0){
              throw new ArithmeticException();//主动地抛出异常,一般在方法中使用
          }
          System.out.println(a/b);
      }
  }

假设这方法中,处理不了这个异常,则在方法上抛出异常(抛到更高级)需捕获。

  
  package com.oop.exception;
  ​
  public class Test2 {
      public static void main(String[] args) {
          try {
              new Test2().test(1,0);
          } catch (ArithmeticException e) {
              e.printStackTrace();
          }
      }
      public void test(int a,int b) throws ArithmeticException{
          if (b==0){
              throw new ArithmeticException();//主动地抛出异常,一般在方法中使用
          }
          System.out.println(a/b);
      }
  }

自定义异常

用户自定义异常类,只需继承Exception类即可。在程序中使用自定义异常类,大体可分为以下几个步骤:

  1. 创建自定义异常类。
  2. 在方法中通过throw关键字抛出异常对象。
  3. 如果在当前抛出异常的方法中处理异常,可以使用try-catch语句捕获并处理;否则在方法的声明处通过throws关键字指明要抛出给方法调用者的异常,继续进行下一步操作。
  4. 在出现异常方法的调用者中捕获并处理异常。