一.简介
程序运行时,发生的不被期望的事件,它阻止了程序按照程序员的预期正常执行,这就是异常。异常发生时,是任程序自生自灭,立刻退出终止,还是输出错误给用户?或者用C语言风格:用函数返回值作为执行状态?。
二. 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 *****************************************/
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 }
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 }
note:语句块中只可以有一个finally,但是可以嵌套多个语句块,每个语句块又包含一个finally。
2.需要注意的地方
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 处理异常
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 }
五. 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 }
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 }
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 }
下面是一个例子,演示了异常的链化:从命令行输入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 */
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)
这就不是一个链化的异常了,仅仅是一个异常追踪栈。
异常链化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 }
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)
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 }
1 发算术异常: java.lang.ArithmeticException: / by zero 2 手动抛出链试异常 : java.lang.NumberFormatException
八. 自定义异常
如果要自定义异常类,则扩展Exception类即可,因此这样的自定义异常都属于检查异常(checked exception)。如果要自定义非检查异常,则扩展自RuntimeException。
按照国际惯例,自定义的异常应该总是包含如下的构造函数:
- 一个无参构造函数
- 一个带有String参数的构造函数,并传递给父类的构造函数。
- 一个带有String参数和Throwable参数,并都传递给父类构造函数
- 一个带有Throwable 参数的构造函数,并传递给父类的构造函数。
1 public class IOException extends Exception 2 { 3 static final long serialVersionUID = 7818375828146090155L; 4 5 public IOException() 6 { 7 super(); 8 } 9 10 public IOException(String message) 11 { 12 super(message); 13 } 14 15 public IOException(String message, Throwable cause) 16 { 17 super(message, cause); 18 } 19 20 21 public IOException(Throwable cause) 22 { 23 super(cause); 24 } 25 }
九. 异常的注意事项
1、当子类重写父类的带有 throws声明的函数时,其throws声明的异常必须在父类异常的可控范围内——用于处理父类的throws方法的异常处理器,必须也适用于子类的这个带throws方法 。这是为了支持多态。
例如,父类方法throws 的是2个异常,子类就不能throws 3个及以上的异常。父类throws IOException,子类就必须throws IOException或者IOException的子类。
至于为什么?我想,也许下面的例子可以说明:
1 public class Parent { 2 public void a() throw AException { 3 } 4 } 5 public class Child extends Parent { 6 public void a() throw AException, BException { 7 } 8 } 9 10 public class Test { 11 public void main(String[] args) { 12 Parent p = new Child(); 13 try { 14 p.a();//这里在调用者就不知道抛出BException了!这样就会导致一些问题 15 } catch(AException e) {//只能补货A,但实际上可能发生了B异常 16 doSomething(); 17 } 18 } 19 }
2、Java程序可以是多线程的。每一个线程都是一个独立的执行流,独立的函数调用栈。如果程序只有一个线程,那么没有被任何代码处理的异常 会导致程序终止。如果是多线程的,那么没有被任何代码处理的异常仅仅会导致异常所在的线程结束。
编译错误:Unreachable catch block for SQLException. It is already handled by the catch block for Exception
十.finally块和return----(可能会造成异常丢失等现象)
首先一个不容易理解的事实:在 try块中即便有return,break,continue等改变执行流的语句,finally也会执行。
1 public static void main(String[] args) 2 { 3 int re = bar(); 4 System.out.println(re); 5 } 6 private static int bar() 7 { 8 try{ 9 return 5; 10 } finally{ 11 System.out.println("finally"); 12 } 13 } 14 /*输出: 15 finally 16 5 17 */
1. finally中的return 会覆盖 try 或者catch中的返回值。
1 public static void main(String[] args) 2 { 3 int result; 4 5 result = foo(); 6 System.out.println(result); /////////2 7 8 result = bar(); 9 System.out.println(result); /////////2 10 } 11 12 @SuppressWarnings("finally") 13 public static int foo() 14 { 15 trz{ 16 int a = 5 / 0; 17 } catch (Exception e){ 18 return 1; 19 } finally{ 20 return 2; 21 } 22 23 } 24 25 @SuppressWarnings("finally") 26 public static int bar() 27 { 28 try { 29 return 1; 30 }finally { 31 return 2; 32 } 33 }
2.finally中的return会抑制(消灭)前面try或者catch块中的异常
1 class TestException 2 { 3 public static void main(String[] args) 4 { 5 int result; 6 try{ 7 result = foo(); 8 System.out.println(result); //输出100 9 } catch (Exception e){ 10 System.out.println(e.getMessage()); //没有捕获到异常 11 } 12 13 14 try{ 15 result = bar(); 16 System.out.println(result); //输出100 17 } catch (Exception e){ 18 System.out.println(e.getMessage()); //没有捕获到异常 19 } 20 } 21 22 //catch中的异常被抑制 23 @SuppressWarnings("finally") 24 public static int foo() throws Exception 25 { 26 try { 27 int a = 5/0; 28 return 1; 29 }catch(ArithmeticException amExp) { 30 throw new Exception("我将被忽略,因为下面的finally中使用了return"); 31 }finally { 32 return 100; 33 } 34 } 35 36 //try中的异常被抑制 37 @SuppressWarnings("finally") 38 public static int bar() throws Exception 39 { 40 try { 41 int a = 5/0; 42 return 1; 43 }finally { 44 return 100; 45 } 46 } 47 }
3.finally中的异常会覆盖(消灭)前面try或者catch中的异常
1 class TestException 2 { 3 public static void main(String[] args) 4 { 5 int result; 6 try{ 7 result = foo(); 8 } catch (Exception e){ 9 System.out.println(e.getMessage()); //输出:我是finaly中的Exception 10 } 11 12 13 try{ 14 result = bar(); 15 } catch (Exception e){ 16 System.out.println(e.getMessage()); //输出:我是finaly中的Exception 17 } 18 } 19 20 //catch中的异常被抑制 21 @SuppressWarnings("finally") 22 public static int foo() throws Exception 23 { 24 try { 25 int a = 5/0; 26 return 1; 27 }catch(ArithmeticException amExp) { 28 throw new Exception("我将被忽略,因为下面的finally中抛出了新的异常"); 29 }finally { 30 throw new Exception("我是finaly中的Exception"); 31 } 32 } 33 34 //try中的异常被抑制 35 @SuppressWarnings("finally") 36 public static int bar() throws Exception 37 { 38 try { 39 int a = 5/0; 40 return 1; 41 }finally { 42 throw new Exception("我是finaly中的Exception"); 43 } 44 45 } 46 }
上面的3个例子都异于常人的编码思维,因此我建议:
- 不要在fianlly中使用return。
- 不要在finally中抛出异常。
- 减轻finally的任务,不要在finally中做一些其它的事情,finally块仅仅用来释放资源是最合适的。
- 将尽量将所有的return写在函数的最后面,而不是try ... catch ... finally中
十一. Throw 和 Throws的区别
(1)throw是放在函数体内,而throws是函数声明中
(2)throw是异常抛出,且抛出的是对象;throws是异常声明,且声明的是异常的种类
(3)函数中throw抛出的对象只能有一个,但是throws声明的异常种类却可以有多个。因为throw一旦抛出了一个异常对象后,就会中断当前程序的执行,因此编译器是不允许 throw多个对象的。
(4)如果throw了一个非RuntimeException,必须在函数声明处抛出异常或者在本函数中进行异常处理。
十一. 异常处理的优缺点
优点:
1. 业务代码和错误处理代码分离.
2. 强制程序猿考虑程序的稳定性.
3. 有利于代码调试(异常信息)
4. 即使任何异常产生, 能保证占用的释放(finally)
缺点:
1. 异常嵌套难免影响代码可读性
2. 并不能令程序逻辑更加清晰.
3. 异常并不能解决所有问题
参考文献:
1.https://www.cnblogs.com/lulipro/p/7504267.html
2.https://www.cnblogs.com/taiwan/p/7073743.html
3.https://blog.csdn.net/chaplinlong/article/details/50983594