JavaEE-19-Java异常处理

一.异常定义:

在Java中用类的形式对不正常情况进行了描述和封装对象,其中,描述不正常情况的类,就称为异常类。

以前,正常流程代码和问题处理代码相结合,现在将正常流程代码和问题处理代码分离,提高阅读性。

其实,异常就是Java通过面向对象的思想将问题封装成了对象。用异常类对其进行描述。

不同的问题用不同的类进行具体的描述。(比如角标越界、空指针等等)

问题很多,意味着描述的类也很多,将其共性进行向上提取,形成了异常体系。

最终问题(不正常情况)就分成了两大类:

  1.一般不可处理的。(Error)

  2.可以处理的。(Exception)

 

二.异常的体系:

  Throwable:无论是Error,还是Exception,都是问题,问题发生就应该抛出,让调用者知道并处理。

  该体系的特点就在于Throwable及其所有的子类都具有可抛性。

    Error:(不可处理的)

      通常出现重大问题如:运行的类不存在或者内存溢出等。(注意,是运行时的异常,而编译时是没有错误提示的)

      特点:是由jvm抛出的严重性的问题,这种问题发生一般不针对性处理,直接修改程序。

    Exception:(可处理的)

      在运行时运行出现的一起情况,可以通过try catch finaly。

Exception和Error的子类名都是以父类名作为后缀。

该体系的特点就在于Throwable及其所有的子类都具有可抛性。

 

三.异常的分类:

  1.编译时被检测异常:只要是Exception和其子类都是,除了特殊子类RuntimeException体系。

    这种问题一旦出现,Java程序必须显式处理,否则程序就会发生错误,无法通过编译。

    (这样的问题都可以针对性处理)

  2.编译时不检测异常(运行时异常):就是Exception中的RuntimeException和其子类。

    这种问题的发生,无法让功能继续,运算无法进行,更多是因为调用者的原因导致的或者引发了内部状态的改变导致的。

    (这种问题无需显式处理,直接编译通过,在运行时,让调用者调用时的程序强制停止,让调用者对代码进行修正。)

所以自定义异常时,要么继承Exception,要么继承RuntimeException。

注:RuntimeException是那些可能在Java虚拟机正常运行期间抛出的异常的超类。

 

四.异常处理的两种方式:

  选择原则:如果该功能内部可以将问题处理,则用try...catch;(后续程序需要继续运行)---一般会使用这种方式

       如果内部处理不了,则交由调用者来处理,使用throw抛出异常。(后续程序不需要继续运行)

1. throw 抛出异常

  throws和throw的区别:

  1.throws使用在方法声明后面;

   throw使用在方法体内。

  2.throws抛出的是异常类,可以抛出多个,用逗号隔开;

   throw抛出的是对象,一次只能抛出一个。

  3.throws表示抛出异常,由该方法的调用者来处理;

   throw表示抛出异常,由方法体内的语句处理;

  4.throws表示出现异常的一种可能性,并不一定会发生这些异常

   而throw则表示一定抛出了某种异常。

 

2. try...catch...finally

  异常处理的捕捉形式:这是可以对异常进行针对性处理的方式,前面的异常不处理,后面的代码将不会继续运行。

  原则:

    1.try里面的代码越少越好(try中的代码需要走异常处理机制,即java虚拟机需要开辟新的资源来管理它)

    2.catch中的代码必须有内容,哪怕是给个简单的提示,也能让人知道异常的原因是什么,不然就是隐藏异常了。

  具体格式:

    try {

      //可能出现异常的代码

      }

    catch(异常类 变量) {

      //处理异常的代码

      }

    finnaly{

      //一般会被执行的代码

      }

 举例:

 1 public class ExceptionDemo {
 2     public static void main(String[] args) {
 3         method();
 4     }
 5     private static void method() {
 6         int a = 10;
 7         int b = 0;
 8         try {
 9             System.out.println(a / b);
10         } catch (Exception e) {
11             System.out.println("除数不能为0");
12         }
13 
14         int arr[] = { 1, 2, 3 };
15         try {
16             System.out.println(arr[3]);
17         } catch (Exception e) {
18             System.out.println("越界");
19         }
20         System.out.println("over");
21     }
22 }

上面这段代码有个问题,但代码中有多个异常时,多次使用try语句,这样既繁琐,代码也不美观。

此时,我们可以对代码进行改进,将有异常的代码放到同一个try代码块中,具体如下:

注意:两个catch的参数的类型不能相同。

public class ExceptionDemo {
    public static void main(String[] args) {
        method();
    }
    private static void method() {
        int a = 10;
        int b = 0;
        int arr[] = { 1, 2, 3 };
        try {
            System.out.println(a / b);
            System.out.println(arr[3]);
        } catch (ArithmeticException e) {
            System.out.println("除数不能为0");
        } catch (ArrayIndexOutOfBoundsException e) {
            System.out.println("越界");
        }
        System.out.println("over");
    }
}

 上面的代码中,一个try对应多个catch的方式相对于第一个代码,是变得简洁了些。

但JDK7之后,出现了一个更简洁的方式:catch中使用多个异常名,这样一来,catch也只要一个就够了。

注意:这种方式有两个弊端

    1.catch语句是同一条,即处理方式是一样的;

    2.异常名不能使用exception;

  格式:

  try{

    ......

  }catch(异常名1 | 异常名2 | ...... 变量) {

    ......

  } 

 

异常处理的原则:

  1.函数内容如果抛出需要检测的异常,那么函数上必须要声明。

   否则必须在函数内用trycatch捕捉,否则编译失败。

  2.如果调用到了声明异常的函数,要么trycatch要么throws,否则编译失败。

  3.什么时候catch,什么时候throws呢?

   功能内容可以解决,用catch;

   解决不了,用throws告诉调用者,由调用者解决。

  4.一个功能如果抛出了多个异常,那么调用时,必须有对应多个catch进行针对性的处理。

   内部有几个需要检测的异常,就抛几个异常,抛出几个,就catch几个。

 

try catch finally组合特点:

  1.try catch finally

  2.try catch(多个)  当没有必要资源需要释放时,可以不用定义finally。

  3.try finally 异常无法直接catch处理,但要资源需要关闭。

 

举例:  

  毕老师用电脑讲课:
    对象:毕老师,电脑
    方法:讲课,运行

    异常1:电脑蓝屏,异常2:电脑冒烟

  代码:

 1 class LanPingException extends Exception//异常1
 2 {
 3     LanPingException(String msg)
 4     {
 5         super(msg);
 6     }
 7 }
 8 class MaoYanException extends Exception//异常2
 9 {
10     MaoYanException(String msg)
11     {
12         super(msg);
13     }
14 }
15 class NoPlanException extends Exception//异常转换
16 {
17     NoPlanException(String msg)
18     {
19         super(msg);
20     }
21 }
22 
23 class Computer//对象1
24 {
25     private int state = 2;//设定电脑开机后的3种状态
26     public void run() throws LanPingException,MaoYanException
27     {
28         if(state==1)
29             throw new LanPingException("电脑蓝屏啦!");
30         if(state==2)
31             throw new MaoYanException("电脑冒烟啦!");
32         System.out.println("电脑运行");
33     }
34     public void reset()
35     {
36         state = 0;
37         System.out.println("电脑重启");
38     }
39 }
40 
41 class Teacher//对象2
42 {
43     private String name;
44     private Computer comp;
45     Teacher(String name)
46     {
47         this.name = name;
48         comp = new Computer();
49     }
50 
51     public void prelect() throws NoPlanException
52     {
53         try
54         {
55             comp.run();
56             System.out.println(name+"讲课");
57         }
58         catch(LanPingException e)
59         {
60             System.out.println(e.toString());//显示错误异常信息
61             comp.reset();
62             prelect();
63         }
64         catch(MaoYanException e)
65         {
66             System.out.println(e.toString());
67             test();
68             throw new NoPlanException("课时进度无法完成,原因:"+e.getMessage());
69         }
70         
71     }
72     public void test()
73     {
74         System.out.println("大家练习");
75     }
76 }
77 
78 class ExceptionTest 
79 {
80     public static void main(String[] args) 
81     {
82         Teacher t = new Teacher("毕老师");
83         try
84         {
85             t.prelect();
86         }
87         catch (NoPlanException e)
88         {
89             System.out.println(e.toString()+"...");
90             System.out.println("换人");
91         }
92         
93     }
94 }
View Code

 

异常的注意事项:

  1.子类在覆盖父类方法时,父类的方法如果抛出了异常,那么子类的方法只能抛出父类的异常或者该异常的子类。

  2.如果父类抛出多个异常,那么子类只能抛出父类异常的子集。

  用一句话概括上面的话就是:子类覆盖父类只能抛出父类的异常或者子类或者子集。

  注意:如果父类的方法没有抛出异常,那么子类覆盖时绝对不能抛。

 

posted @ 2019-06-25 15:47  benon  阅读(307)  评论(0编辑  收藏  举报