05. Java基础之异常

Posted Hermioner

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了05. Java基础之异常相关的知识,希望对你有一定的参考价值。

一.简介

程序运行时,发生的不被期望的事件,它阻止了程序按照程序员的预期正常执行,这就是异常。异常发生时,是任程序自生自灭,立刻退出终止,还是输出错误给用户?或者用C语言风格:用函数返回值作为执行状态?。

 
Java提供了更加优秀的解决办法:异常处理机制。
 
异常处理机制能让程序在异常发生时,按照代码的预先设定的异常处理逻辑,针对性地处理异常,让程序尽最大可能恢复正常并继续执行,且保持代码的清晰。
Java中的异常可以是函数中的语句执行时引发的,也可以是程序员通过throw 语句手动抛出的,只要在Java程序中产生了异常,就会用一个对应类型的异常对象来封装异常,JRE就会试图寻找异常处理程序来处理异常。
 
Throwable类是Java异常类型的顶层父类,一个对象只有是 Throwable 类的(直接或者间接)实例,他才是一个异常对象,才能被异常处理机制识别。JDK中内建了一些常用的异常类,我们也可以自定义异常。

二. Java异常的分类和类结构图

Java标准库内建了一些通用的异常,这些类以Throwable为顶层父类。

Throwable又派生出Error类和Exception类。

错误:Error类以及他的子类的实例,代表了JVM本身的错误。错误不能被程序员通过代码处理,Error很少出现。因此,程序员应该关注Exception为父类的分支下的各种异常类。

异常:Exception以及他的子类,代表程序运行时发送的各种不期望发生的事件。可以被Java异常处理机制使用,是异常处理的核心。

 

总体上我们根据Javac对异常的处理要求,将异常类分为2类。

非检查异常(unckecked exception):Error 和 RuntimeException 以及他们的子类。javac在编译时,不会提示和发现这样的异常,不要求在程序处理这些异常。所以如果愿意,我们可以编写代码处理(使用try...catch...finally)这样的异常,也可以不处理。对于这些异常,我们应该修正代码,而不是去通过异常处理器处理 。这样的异常发生的原因多半是代码写的有问题。如除0错误ArithmeticException,错误的强制类型转换错误ClassCastException,数组索引越界ArrayIndexOutOfBoundsException,使用了空对象NullPointerException等等。

 

检查异常(checked exception):除了Error 和 RuntimeException的其它异常。javac强制要求程序员为这样的异常做预备处理工作(使用try...catch...finally或者throws)。在方法中要么用try-catch语句捕获它并处理,要么用throws子句声明抛出它,否则编译不会通过。这样的异常一般是由程序的运行环境导致的。因为程序可能被运行在各种未知的环境下,而程序员无法干预用户如何使用他编写的程序,于是程序员就应该为这样的异常时刻准备着。如SQLException , IOException,ClassNotFoundException 等。

需要明确的是:检查和非检查是对于javac来说的,这样就很好理解和区分了。

三. 初识异常

1. 未受检异常

 1 package com.example;
 2 import java. util .Scanner ;
 3 public class AllDemo
 4 {
 5       public static void main (String [] args )
 6       {
 7             System . out. println( "----欢迎使用命令行除法计算器----" ) ;
 8             CMDCalculate ();
 9       }
10       public static void CMDCalculate ()
11       {
12             Scanner scan = new Scanner ( System. in );
13             int num1 = scan .nextInt () ;
14             int num2 = scan .nextInt () ;
15             int result = devide (num1 , num2 ) ;
16             System . out. println( "result:" + result) ;
17             scan .close () ;
18       }
19       public static int devide (int num1, int num2 ){
20             return num1 / num2 ;
21       }
22 }
23 /*****************************************
24 
25 ----欢迎使用命令行除法计算器----
26 1
27 0
28 Exception in thread "main" java.lang.ArithmeticException : / by zero
29      at com.example.AllDemo.devide( AllDemo.java:30 )
30      at com.example.AllDemo.CMDCalculate( AllDemo.java:22 )
31      at com.example.AllDemo.main( AllDemo.java:12 )
32 
33 ----欢迎使用命令行除法计算器----
34 1
35 r
36 Exception in thread "main" java.util.InputMismatchException
37      at java.util.Scanner.throwFor( Scanner.java:864 )
38      at java.util.Scanner.next( Scanner.java:1485 )
39      at java.util.Scanner.nextInt( Scanner.java:2117 )
40      at java.util.Scanner.nextInt( Scanner.java:2076 )
41      at com.example.AllDemo.CMDCalculate( AllDemo.java:20 )
42      at com.example.AllDemo.main( AllDemo.java:12 )
43 *****************************************/
View Code

note:异常信息的格式,先告诉是什么异常,然后再告诉是哪里"at"出现了异常。

异常是在执行某个函数时引发的,而函数又是层级调用,形成调用栈的,因为,只要一个函数发生了异常,那么他的所有的caller都会被异常影响。当这些被影响的函数以异常信息输出时,就形成的了异常追踪栈

异常最先发生的地方,叫做异常抛出点

 

 

从上面的例子可以看出,当devide函数发生除0异常时,devide函数将抛出ArithmeticException异常,因此调用他的CMDCalculate函数也无法正常完成,因此也发送异常,而CMDCalculate的caller——main 因为CMDCalculate抛出异常,也发生了异常,这样一直向调用栈的栈底回溯。这种行为叫做异常的冒泡异常的冒泡是为了在当前发生异常的函数或者这个函数的caller中找到最近的异常处理程序。由于这个例子中没有使用任何异常处理机制,因此异常最终由main函数抛给JRE,导致程序终止。

上面的代码不使用异常处理机制,也可以顺利编译,因为2个异常都是非检查异常。但是下面的例子就必须使用异常处理机制,因为异常是检查异常。

代码中我选择使用throws声明异常,让函数的调用者去处理可能发生的异常。但是为什么只throws了IOException呢?因为FileNotFoundException是IOException的子类,在处理范围内。

2. 受检异常

 1 public void testException() throws IOException
 2 {
 3     //FileInputStream的构造函数会抛出FileNotFoundException
 4     FileInputStream fileIn = new FileInputStream("E:\\\\a.txt");
 5     
 6     int word;
 7     //read方法会抛出IOException
 8     while((word =  fileIn.read())!=-1) 
 9     {
10         System.out.print((char)word);
11     }
12     //close方法会抛出IOException
13     fileIn.clos
14 }
View Code

note:如果方法中没有throws来声明异常,代码将会出现编译错误;如果不想使用throws来声明会出现的异常,也可以再代码中使用try-catch进行异常捕获处理。

四.异常处理的基本语法

在编写代码处理异常时,对于检查异常,有2种不同的处理方式:使用try...catch...finally语句块处理它。或者,在函数签名中使用throws 声明交给函数调用者caller去解决。

1.try...catch...finally语句块

 1 try{
 2      //try块中放可能发生异常的代码。
 3      //如果执行完try且不发生异常,则接着去执行finally块和finally后面的代码(如果有的话)。
 4      //如果发生异常,则尝试去匹配catch块。
 5 
 6 }catch(SQLException SQLexception){
 7     //每一个catch块用于捕获并处理一个特定的异常,或者这异常类型的子类。Java7中可以将多个异常声明在一个catch中。
 8     //catch后面的括号定义了异常类型和异常参数。如果异常与之匹配且是最先匹配到的,则虚拟机将使用这个catch块来处理异常。
 9     //在catch块中可以使用这个块的异常参数来获取异常的相关信息。异常参数是这个catch块中的局部变量,其它块不能访问。
10     //如果当前try块中发生的异常在后续的所有catch中都没捕获到,则先去执行finally,然后到这个函数的外部caller中去匹配异常处理器。
11     //如果try中没有发生异常,则所有的catch块将被忽略。
12 
13 }catch(Exception exception){
14     //...
15 }finally{
16    
17     //finally块通常是可选的。
18    //无论异常是否发生,异常是否匹配被处理,finally都会执行。
19    //一个try至少要有一个catch块,否则, 至少要有1个finally块。但是finally不是用来处理异常的,finally不会捕获异常。
20   //finally主要做一些清理工作,如流的关闭,数据库连接的关闭等。 
21 }
View Code

note:语句块中只可以有一个finally,但是可以嵌套多个语句块,每个语句块又包含一个finally。

2.需要注意的地方

1、try块中的局部变量和catch块中的局部变量(包括异常变量),以及finally中的局部变量,他们之间不可共享使用。(note:说明作用于只在{}之间)
 
2、每一个catch块用于处理一个异常。异常匹配是按照catch块的顺序从上往下寻找的,只有第一个匹配的catch会得到执行。匹配时,不仅运行精确匹配,也支持父类匹配,因此,如果同一个try块下的多个catch异常类型有父子关系,应该将子类异常放在前面,父类异常放在后面,这样保证每个catch块都有存在的意义。
 
3、java中,异常处理的任务就是将执行控制流从异常发生的地方转移到能够处理这种异常的地方去也就是说:当一个函数的某条语句发生异常时,这条语句的后面的语句不会再执行,它失去了焦点。执行流跳转到最近的匹配的异常处理catch代码块去执行,异常被处理完后,执行流会接着在“处理了这个异常的catch代码块”后面接着执行。
有的编程语言当异常被处理后,控制流会恢复到异常抛出点接着执行,这种策略叫做:resumption model of exception handling恢复式异常处理模式
而Java则是让执行流恢复到处理了异常的catch块后接着执行,这种策略叫做:termination model of exception handling终结式异常处理模式
 1 public static void main(String[] args){
 2         try {
 3             foo();
 4         }catch(ArithmeticException ae) {
 5             System.out.println("处理异常");
 6         }
 7 }
 8 public static void foo(){
 9         int a = 5/0;  //异常抛出点
10         System.out.println("为什么还不给我涨工资!!!");  //////////////////////不会执行
11 }
12 
13 output:
14 处理异常
View Code

3.throws函数声明

throws声明:如果一个方法内部的代码会抛出检查异常(checked exception),而方法自己又没有完全处理掉,则javac保证你必须在方法的签名上使用throws关键字声明这些可能抛出的异常,否则编译不通过。

throws是另一种处理异常的方式,它不同于try...catch...finally,throws仅仅是将函数中可能出现的异常向调用者声明,而自己则不具体处理。

采取这种异常处理的原因可能是:方法本身不知道如何处理这样的异常,或者说让调用者处理更好,调用者需要为可能发生的异常负责。

1 public void foo() throws ExceptionType1 , ExceptionType2 ,ExceptionTypeN
2 { 
3      //foo内部可以抛出 ExceptionType1 , ExceptionType2 ,ExceptionTypeN 类的异常,或者他们的子类的异常对象。
4 }
View Code

五. finally块

finally块不管异常是否发生,只要对应的try执行了,则它一定也执行。只有一种方法让finally块不执行:System.exit()。因此finally块通常用来做资源释放操作:关闭文件,关闭数据库连接等等。

良好的编程习惯是:在try块中打开资源,在finally块中清理释放这些资源。

需要注意的地方:

1、finally块没有处理异常的能力。处理异常的只能是catch块

2、在同一try...catch...finally块中 ,如果try中抛出异常,且有匹配的catch块,则先执行catch块,再执行finally块。如果没有catch块匹配,则先执行finally,然后去外面的调用者中寻找合适的catch块。

3、在同一try...catch...finally块中 ,try发生异常,且匹配的catch块中处理异常时也抛出异常,那么后面的finally也会执行:首先执行finally块,然后去外围调用者中寻找合适的catch块。

这是正常的情况,但是也有特例。关于finally有很多恶心,偏、怪、难的问题(比如异常丢失),参考后面的finally块和return部分。

 六. throw异常抛出语句

throw exceptionObject

程序员也可以通过throw语句手动显式的抛出一个异常。throw语句的后面必须是一个异常对象。

throw 语句必须写在函数中,执行throw 语句的地方就是一个异常抛出点,它和由JRE自动形成的异常抛出点没有任何差别。

1 public void save(User user)
2 {
3       if(user  == null) 
4           throw new IllegalArgumentException("User对象为空");
5       //......
6         
7 }
View Code

note:throw跟受检不受检没有关系,它只是抛出异常。相当于try语句中产生了一个异常。

Note:throw和throws使用注意:假设throw了一个新的异常,会编译错误,必须用throws或者在该方法中用try --catch代码块来处理

Question1:为什么throw new Exception()就会错呢?然后new RuntimeException()以及它的子类是不会错的。受检异常是会报错的。

Question2:为什么在catch中直接throw e就不会出错,不需要throws或者异常处理了呢?

Question3:摘自https://www.javatpoint.com/throws-keyword-and-difference-between-throw-and-throws

Which exception should be declared

Ans) checked exception only, because:(只有受检异常需要声明)

  • unchecked Exception: under your control so correct your code.
  • error: beyond your control e.g. you are unable to do anything if there occurs VirtualMachineError or StackOverflowError

Question4: throws可以传播异常

 1 package b;
 2 
 3 import java.io.IOException;
 4 
 5 public class Test {
 6     void m() throws IOException {
 7         throw new IOException("device error");
 8     }
 9 
10     void n() throws IOException {// 必须要有throws,因为throws异常会传播,调用了m(),m()中的异常会传播
11         m();
12     }
13 
14     void p() {
15         try {
16             n();
17         } catch (IOException e) {
18             System.out.println("exception handled");
19         }
20     }
21 
22     public static void main(String args[]) {
23         Test test = new Test();
24         test.p();
25         System.out.println("normal flow...");
26     }
27 }
View Code

exception handled
normal flow...

七. 异常的链化

在一些大型的,模块化的软件开发中,一旦一个地方发生异常,则如骨牌效应一样,将导致一连串的异常。假设B模块完成自己的逻辑需要调用A模块的方法,如果A模块发生异常,则B也将不能完成而发生异常,但是B在抛出异常时,会将A的异常信息掩盖掉,这将使得异常的根源信息丢失。异常的链化可以将多个模块的异常串联起来,使得异常信息不会丢失。

异常链化:以一个异常对象为参数构造新的异常对象。新的异对象将包含先前异常的信息。这项技术主要是异常类的一个带Throwable参数的函数来实现的。这个当做参数的异常,我们叫他根源异常(cause)。

查看Throwable类源码,可以发现里面有一个Throwable字段cause,就是它保存了构造时传递的根源异常参数。这种设计和链表的结点类设计如出一辙,因此形成链也是自然的了。

 1 public class Throwable implements Serializable {
 2     private Throwable cause = this;
 3    
 4     public Throwable(String message, Throwable cause) {
 5         fillInStackTrace();
 6         detailMessage = message;
 7         this.cause = cause;
 8     }
 9      public Throwable(Throwable cause) {
10         fillInStackTrace();
11         detailMessage = (cause==null ? null : cause.toString());
12         this.cause = cause;
13     }
14     
15     //........
16 } 
View Code

下面是一个例子,演示了异常的链化:从命令行输入2个int,将他们相加,输出。输入的数不是int,则导致getInputNumbers异常,从而导致add函数异常,则可以在add函数中抛出

一个链化的异常。

 1 public static void main(String[] args)
 2 {
 3     
 4     System.out.println("请输入2个加数");
 5     int result;
 6     try
 7     {
 8         result = add();
 9         System.out.println("结果:"+result);
10     } catch (Exception e){
11         e.printStackTrace();
12     }
13 }
14 //获取输入的2个整数返回
15 private static List<Integer> getInputNumbers()
16 {
17     List<Integer> nums = new ArrayList<>();
18     Scanner scan = new Scanner(System.in);
19     try {
20         int num1 = scan.nextInt();
21         int num2 = scan.nextInt();
22         nums.add(new Integer(num1));
23         nums.add(new Integer(num2));
24     }catch(InputMismatchException immExp){
25         throw immExp;
26     }finally {
27         scan.close();
28     }
29     return nums;
30 }
31 
32 //执行加法计算
33 private static int add() throws Exception
34 {
35     int result;
36     try {
37         List<Integer> nums =getInputNumbers();
38         result = nums.get(0)  + nums.get(1);
39     }catch(InputMismatchException immExp){
40         throw new Exception("计算失败",immExp);  /////////////////////////////链化:以一个异常对象为参数构造新的异常对象。
41     }
42     return  result;
43 }
44 
45 /*
46 请输入2个加数
47 r 1
48 java.lang.Exception: 计算失败
49     at practise.ExceptionTest.add(ExceptionTest.java:53)
50     at practise.ExceptionTest.main(ExceptionTest.java:18)
51 Caused by: java.util.InputMismatchException
52     at java.util.Scanner.throwFor(Scanner.java:864)
53     at java.util.Scanner.next(Scanner.java:1485)
54     at java.util.Scanner.nextInt(Scanner.java:2117)
55     at java.util.Scanner.nextInt(Scanner.java:2076)
56     at practise.ExceptionTest.getInputNumbers(ExceptionTest.java:30)
57     at practise.ExceptionTest.add(ExceptionTest.java:48)
58     ... 1 more
59 
60 */
View Code

note:倘若注释掉 “ throw new Exception("计算失败",immExp);  /////////////////////////////链化:以一个异常对象为参数构造新的异常对象”,将会得到如下信息:

 1 请输入2个加数
 2 1
 3 r
 4 java.util.InputMismatchException
 5     at java.util.Scanner.throwFor(Unknown Source)
 6     at java.util.Scanner.next(Unknown Source)
 7     at java.util.Scanner.nextInt(Unknown Source)
 8     at java.util.Scanner.nextInt(Unknown Source)
 9     at b.AllDemo.getInputNumbers(AllDemo.java:29)
10     at b.AllDemo.add(AllDemo.java:45)
11     at b.AllDemo.main(AllDemo.java:16)
View Code

 这就不是一个链化的异常了,仅仅是一个异常追踪栈

异常链化eg:

1.

 1 public class MyException  extends Exception{
 2 
 3     public MyException(){
 4         super();
 5 
 6     }
 7     public MyException(String message){
 8         super(message);
 9 
10     }
11     public MyException(Throwable cause){
12         super(cause);
13 
14     }
15 
16     public MyException(String message,Throwable cause){
17         super(message,cause);
18 
19     }
20     public static void main(String[] args) throws MyException {
21         try {
22             throw new Exception("还是好好学习Java");
23         } catch (Exception e) {
24             throw new MyException("坚持就是胜利!",e);
25         }
26     }
27 }
View Code
1 Exception in thread "main" cn.defineException.MyException: 坚持就是胜利!
2     at cn.defineException.MyException.main(MyException.java:26)
3 Caused by: java.lang.Exception: 还是好好学习Java
4     at cn.defineException.MyException.main(MyException.java:24)
View Code

2.

 1 public class Main {
 2     public static void main (String args[])throws Exception  {
 3         int n=20,result=0;
 4         try{
 5             result=n/0;
 6             System.out.println("结果为"+result);
 7         }
 8         catch(ArithmeticException ex){
 9             System.out.println("发算术异常: "+ex);
10             try {
11                 throw new NumberFormatException();
12             }
13             catch(NumberFormatException ex1) {
14                 System.out.println("手动抛出链试异常 : "+ex1);
15             }
16         }
17     }
18 }
View Code
1 发算术异常: java.lang.ArithmeticException: / by zero
2 手动抛出链试异常 : java.lang.NumberFormatException
View Code

 

八. 自定义异常

如果要自定义异常类,则扩展Exception类即可,因此这样的自定义异常都属于检查异常(checked exception)。如果要自定义非检查异常,则扩展自RuntimeException

按照国际惯例,自定义的异常应该总是包含如下的构造函数:

  • 一个无参构造函数
  • 一个带有String参数的构造函数,并传递给父类的构造函数。
  • 一个带有String参数和Throwable参数,并都传递给父类构造函数
  • 一个带有Throwable 参数的构造函数,并传递给父类的构造函数。
 下面是IOException类的完整源代码,可以借鉴