Java异常处理

一.异常的概念

异常:就是程序运行过程中,出现了非正常的情况,异常也是错误的一种,但不是所有错误都是异常,异常会导致JVM非正常停止。

异常有的时候是因为用户错误引起,有的时候是程序错误引起的,还有的一些是其他原因导致的。

二.异常的体系

异常在java当中是以对象来处理的,异常的基类是:java.lang.Throwable

异常基类下有两个子类:java.lang.Errorjava.lang.Exception,即上面所说的错误和异常。

异常体系.png

上图列举了Throwable的主要分支和一些常见的异常。

注意:

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

三.异常的分类

异常的分类,根据在编译时期还是运行时期检查异常可以分为两类:

  1. 编译时期异常(检测异常):在编译时期就会检测,如果没有处理异常,则编译失败。
  2. 运行时期异常(非检测异常):在编译时期不会检测(也不会报错)。

注意:

除了RuntimeException以及其子类外,其他的所有Exception类及其子类都是检测异常,都是需要检查的,当出现这些异常时,要么使用try-catch语句捕获异常,要么用throws语句抛出异常,否则编译无法通过

RuntimeException,这类异常时不检测的,可以处理,但不一定会处理,一般不处理。

四.异常的处理

异常的处理本质上有两个:抛出异常捕获异常

简单点说:

  1. 抛出异常:就是自己不处理,向上抛出异常,由调用者去决定怎么处理。

    比如方法中对参数做检测:

    if (obj == null) {
        throw new NullPointerException();
    }
    
  2. 捕获异常:就是自己捕获异常,并处理,处理捕获异常需要使用try-catch语句(下面会讲到)

五.异常处理的语法

异常处理的语法主要是五个关键字:try catch finally throw throws

try:用于监测,将可能产生异常的代码放到try语句块之中,当异常产生时,异常就会被抛出。

catch:用于捕获异常,catch用来捕获try语句块中抛出的异常。

finally:finally语句块的代码总是会被执行,主要是做一些清理(比如关闭打开了的文件句柄等),当try或catch语句块中有return语句或者throw语句,总是先执行完finally语句块,在返回执行try或catch中的return语句或throw语句,所以如果finally中使用了return或者throw等终止方法的语句,就不会再返回了,方法会直接停止,所以不能再finally中使用return或者throw

throw:用于抛出异常

throws:用于在方法声明中,声明该方法可能抛出的异常

  • 捕获异常处理

    1. try-catch语句处理

      try {
         // 可能发生异常的代码
      } catch (Exception e) {
          // 异常处理
      } catch (Exception e) {
             // 异常处理
      }
      ...
      

      比如:

      public class TryCacthTest {
          public static void main(String[] args) {
              try {  // 当方法中产生异常时,必须要有处理,要么捕获,要么声明抛出
                  read("a.txt");
              } catch (FileNotFoundException e) {
                  System.out.println(e);
              }
          }
          // 有异常,编译期异常,检测异常
          public static void read(String path) {
              if (!path.equals("a.txt")) {   // 假设文件名不是a.txt,认为是错误,抛出异常
                  throw new FileNotFoundException("文件不存在");
              }
          }
      }
      

      如何获取异常信息?

      答:在Throwable类中定义了一些查看方法,有以下三种:

      1. public String getMessage():获取异常的描述信息,错误原因
      2. public String toString():获取异常的类型和异常的描述信息
      3. public String printStackTrace():打印异常的跟着栈信息并输出到控制台

      一般在开发和调试阶段,都得使用printStackTrace,其包含了异常的类型,异常的原因,以及异常出现的位置。

    2. try-catch-catch-...多个catch情况

      有时候一个代码段可以引发多个异常,这种情况可以定义多个catch,对每一种类型的异常进行捕获,当异常发生时候,每个catch语句会依次检查,第一个匹配异常类型的catch语句执行,其他的所有catch语句不会再执行.

      编写多个catch语句的注意事项:子类在前,父类在后。

    3. try-catch-finally

      同try-catch或try-catch-catch-...,唯一的不同就是finally语句块里的无论异常是否发生,都会执行,并且不能在finally中添加return。

  • 抛出异常处理

    1. 使用throw抛出

      具体语法:throw XXXException

      如果throw语句在try块里,则会检查是否有匹配的catch语句,如果发现了有匹配的,则转向对应catch语句;如果没有发现,默认的异常处理程序则会中断程序,并且打印堆栈信息。

      注意:程序执行完throw语句之后立即停止,throw后面的语句不会被执行。

      比如:

      class ThrowTest {
          public static func() {
              try {
                  throw new NullPointerException("Test");
              } catch (NullPointerException e) {
                  throw e;
              }
          }
          
          public static void main(String[] args) {
              try {
                  func();
              } catch (NullPointerException e) {
                  e.printStackTrace();
              }
          }
      }
      /*
         程序中两次处理了异常,第一次,在main中调用func函数时候,抛出了异常,但func函数的try-catch捕获了这个异常,在catch里又一次抛出了异常,给到了调用者main函数,main里的try-catch进行了捕获,最后调用printStackTrace()
      */
      
    2. throws声明异常

      如果一个方法可能产生一个异常,但本身不处理它,则必须在方法声明中使用throws声明异常,否则导致编译错误。

      使用方法如下,如果存在多个异常,则用逗号隔开

      public void func() throws Exception1,Exception2,...

      注意:

      关于throws抛出异常的规则如下:

      • 如果是不检测的异常,即ErrorRuntimeException或其子类,可以不用throws声明要抛出异常,也可以不用try-catch捕获,编译仍可以通过,但在运行时会被系统抛出。
      • 对于检测的异常,要么用try-catch语句捕获,要么用throw抛出异常,再用throws声明异常,否则会导致编译错误。
      • 当抛出了异常给调用者,调用者也无法处理异常时,必须要继续抛出,不能置之不理。
      • 如果要覆盖一个方法,则声明的任何异常必须是被覆盖方法所声明的异常的同类或子类。

六.自定义异常

除了使用java内置的异常类外,还可以自定义异常,用户自定义异常,只需要继承Exception类即可

自定义异常,可以分为以下几个步骤:

  1. 创建自定义异常类
  2. 在方法中通过throw抛出异常对象
  3. 如果在当前抛出异常的方法中处理异常,可以使用try-catch捕获并处理;否则在方法的声明使用throws声明要抛出的异常
  4. 在异常出现的方法中捕获并处理异常

比如:模拟注册操作,用户名存在,则抛出异常并提示

// 自定义异常
public class RegisterException extends Exception {
    // 空参构造函数
    public RegisterException() {
        
    }
    // 带参构造函数,message表示异常提示
    public RegisterException(String message) {
        super(message);
    }
}
public class Demo {
    private static String[] names = {"Bill", "hello", "cddd"};
    
    public static void main(String[] args) {
        try {
            // 可能出现异常的代码
            checkUsername("nill");
            System.out.println("注册成功");// 如果没有异常就是注册成功的
        } catch (RegisterException e) {
            // 处理异常
            e.printStackTrace();
        }
    }
    public static boolean checkUsername(String uname) throws RegisterException {
        for (String name : names) {
            if (name.equals(uname)) {
                throw new RegisterException(name + "已经被注册");
            }
        }
        return true;
    }
}

七.总结

异常总结.jpg

异常总结1.jpg

参考文档:

Java提高篇——Java 异常处理

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,558评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,002评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,036评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,024评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,144评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,255评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,295评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,068评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,478评论 1 305
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,789评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,965评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,649评论 4 336
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,267评论 3 318
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,982评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,223评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,800评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,847评论 2 351

推荐阅读更多精彩内容

  • 来源:https://www.cnblogs.com/Qian123/p/5715402.html#_label0...
    Alex笔记阅读 374评论 0 1
  • 异常的处理应该先预计到所有可能出现的异常,然后考虑能否完全避免异常,如果不能完全避免,再考虑异常发生时的具体处理方...
    老干妈炒饭阅读 313评论 0 0
  • JAVA 异常类型结构 Error 和 Exeption 受查异常和非受查异常 异常的抛出与捕获 直接抛出异常 封...
    哦00阅读 389评论 0 0
  • 异常分类 Java将异常分为两种,Checked异常和Runtime异常。Java认为Checked异常都是可以在...
    LLorenzo阅读 707评论 0 1
  • 一、异常概述异常指的是在程序运行中出现的错误,在程序执行期间发生的时间,它中断了正在执行的程序的正常指令流。如,执...
    7d4b2866e78f阅读 699评论 0 1