• 那是从何处传来的钟声呢?偶尔听到那钟声,平添一份喜悦与向往之情。

深入理解Java异常处理机制

后端 Nanait 12年前 (2012-07-10) 912次浏览 已收录 0个评论 扫描二维码
文章目录[隐藏]

1. 引子

try…catch…finally 恐怕是大家再熟悉不过的语句了,而且感觉用起来也是很简单,逻辑上似乎也是很容易理解。不过,我亲自体验的“教训”告诉我,这个东西可不是想象中的那么简单、听话。不信?那你看看下面的代码,“猜猜”它执行后的结果会是什么?不要往后看答案、也不许执行代码看真正答案哦。如果你的答案是正确,那么这篇文章你就不用浪费时间看啦。

  1. package Test;
  2. public class TestException {
  3.     public TestException() {
  4.     }
  5.     boolean testEx() throws Exception {
  6.         boolean ret = true;
  7.         try {
  8.             ret = testEx1();
  9.         } catch (Exception e) {
  10.             System.out.println(“testEx, catch exception”);
  11.             ret = false;
  12.             throw e;
  13.         } finally {
  14.             System.out.println(“testEx, finally; return value=” + ret);
  15.             return ret;
  16.         }
  17.     }
  18.     boolean testEx1() throws Exception {
  19.         boolean ret = true;
  20.         try {
  21.             ret = testEx2();
  22.             if (!ret) {
  23.                 return false;
  24.             }
  25.             System.out.println(“testEx1, at the end of try”);
  26.             return ret;
  27.         } catch (Exception e) {
  28.             System.out.println(“testEx1, catch exception”);
  29.             ret = false;
  30.             throw e;
  31.         } finally {
  32.             System.out.println(“testEx1, finally; return value=” + ret);
  33.             return ret;
  34.         }
  35.     }
  36.     boolean testEx2() throws Exception {
  37.         boolean ret = true;
  38.         try {
  39.             int b = 12;
  40.             int c;
  41.             for (int i = 2; i >= –2; i–) {
  42.                 c = b / i;
  43.                 System.out.println(“i=” + i);
  44.             }
  45.             return true;
  46.         } catch (Exception e) {
  47.             System.out.println(“testEx2, catch exception”);
  48.             ret = false;
  49.             throw e;
  50.         } finally {
  51.             System.out.println(“testEx2, finally; return value=” + ret);
  52.             return ret;
  53.         }
  54.     }
  55.     public static void main(String[] args) {
  56.         TestException testException1 = new TestException();
  57.         try {
  58.             testException1.testEx();
  59.         } catch (Exception e) {
  60.             e.printStackTrace();
  61.         }
  62.     }
  63. }

你的答案是什么?是下面的答案吗?

i=2

i=1

testEx2, catch exception

testEx2, finally; return value=false

testEx1, catch exception

testEx1, finally; return value=false

testEx, catch exception

testEx, finally; return value=false

如果你的答案真的如上面所说,那么你错啦。^_^,那就建议你仔细看一看这篇文章或者拿上面的代码按各种不同的情况修改、执行、测试,你会发现有很多事情不是原来想象中的那么简单的。现在公布正确答案:

i=2

i=1

testEx2, catch exception

testEx2, finally; return value=false

testEx1, finally; return value=false

testEx, finally; return value=false

注意说明:

finally 语句块不应该出现 应该出现 return。上面的 return ret 最好是其他语句来处理相关逻辑。

 

 2.Java异常

异常指不期而至的各种状况,如:文件找不到、网络连接失败、非法参数等。异常是一个事件,它发生在程序运行期间,干扰了正常的指令流程。Java通 过 API 中 Throwable 类的众多子类描述各种不同的异常。因而,Java异常都是对象,是 Throwable 子类的实例,描述了出现在一段编码中的 错误条件。当条件生成时,错误将引发异常。

Java异常类层次结构图:

深入理解 Java 异常处理机制

图 1 Java异常类层次结构图

 

Java 中,所有的异常都有一个共同的祖先 Throwable(可抛出)。Throwable 指定代码中可用异常传播机制通过 Java 应用程序传输的任何问题的共性。

Throwable: 有两个重要的子类:Exception(异常)和 Error(错误),二者都是 Java 异常处理的重要子类,各自都包含大量子类。

Error(错误):是程序无法处理的错误,表示运行应用程序中较严重问题。大多数错误与代码编写者执行的操作无关,而表示代码运行时 JVM(Java 虚拟机)出现的问题。例如,Java 虚拟机运行错误(Virtual MachineError),当 JVM 不再有继续执行操作所需的内存资源时,将出现 OutOfMemoryError。这些异常发生时,Java 虚拟机(JVM)一般会选择线程终止。

。这些错误表示故障发生于虚拟机自身、或者发生在虚拟机试图执行应用时,如 Java 虚拟机运行错误(Virtual MachineError)、类定义错误(NoClassDefFoundError)等。这些错误是不可查的,因为它们在应用程序的控制和处理能力之 外,而且绝大多数是程序运行时不允许出现的状况。对于设计合理的应用程序来说,即使确实发生了错误,本质上也不应该试图去处理它所引起的异常状况。在 Java 中,错误通过 Error 的子类描述。

Exception(异常):是程序本身可以处理的异常。

Exception 类有一个重要的子类 RuntimeException。RuntimeException 类及其子类表示“JVM 常用操作”引发的错误。例如,若试图使用空值对象引用、除数为零或数组越界,则分别引发运行时异常(NullPointerException、ArithmeticException)和 ArrayIndexOutOfBoundException。

 

注意:异常和错误的区别:异常能被程序本身可以处理,错误是无法处理。

 

通常,Java 的异常(包括 Exception 和 Error)分为可查的异常(checked exceptions)和不可查的异常(unchecked exceptions)

可查异常(编译器要求必须处置的异常):正确的程序在运行中,很容易出现的、情理可容的异常状况。可查异常虽然是异常状况,但在一定程度上它的发生是可以预计的,而且一旦发生这种异常状况,就必须采取某种方式进行处理。

除了 RuntimeException 及其子类以外,其他的 Exception 类及其子类都属于可查异常。这种异常的特点是 Java 编译器会检查它,也就是说,当程序中可能出现这类异常,要么用 try-catch 语句捕获它,要么用 throws 子句声明抛出它,否则编译不会通过。

不可查异常(编译器不要求强制处置的异常):包括运行时异常(RuntimeException 与其子类)和错误(Error)。

Exception 这种异常分两大类运行时异常和非运行时异常(编译异常)。程序中应当尽可能去处理这些异常。

运行时异常:都是 RuntimeException 类及其子类异常,如 NullPointerException(空指针异常)、IndexOutOfBoundsException(下标越界异常)等,这些异常是不检查异常,程序中可以选择捕获处理,也可以不处理。这些异常一般是由程序逻辑错误引起的,程序应该从逻辑角度尽可能避免这类异常的发生。

运行时异常的特点是 Java 编译器不会检查它,也就是说,当程序中可能出现这类异常,即使没有用 try-catch 语句捕获它,也没有用 throws 子句声明抛出它,也会编译通过。
非运行时异常 (编译异常):是 RuntimeException 以外的异常,类型上都属于 Exception 类及其子类。从程序语法角度讲是必须进行处理的异常,如果不处理,程序就不能编译通过。如 IOException、SQLException 等以及用户自定义的 Exception 异常,一般情况下不自定义检查异常。

 4.处理异常机制

在 Java 应用程序中,异常处理机制为:抛出异常,捕捉异常。

抛出异常:当一个方法出现错误引发异常时,方法创建异常对象并交付运行时系统,异常对象中包含了异常类型和异常出现时的程序状态等异常信息。运行时系统负责寻找处置异常的代码并执行。

捕获异常:在方法抛出异常之后,运行时系统将转为寻找合适的异常处理器(exception handler)。潜在的异常处理器是异常发生时依次存留在调用栈中的方法的集合。当异常处理器所能处理的异常类型与方法抛出的异常类型相符时,即为合适 的异常处理器。运行时系统从发生异常的方法开始,依次回查调用栈中的方法,直至找到含有合适异常处理器的方法并执行。当运行时系统遍历调用栈而未找到合适 的异常处理器,则运行时系统终止。同时,意味着 Java 程序的终止。

对于运行时异常、错误或可查异常,Java 技术所要求的异常处理方式有所不同。

由于运行时异常的不可查性,为了更合理、更容易地实现应用程序,Java 规定,运行时异常将由 Java 运行时系统自动抛出,允许应用程序忽略运行时异常。

对于方法运行中可能出现的 Error,当运行方法不欲捕捉时,Java 允许该方法不做任何抛出声明。因为,大多数 Error 异常属于永远不能被允许发生的状况,也属于合理的应用程序不该捕捉的异常。

对于所有的可查异常,Java 规定:一个方法必须捕捉,或者声明抛出方法之外。也就是说,当一个方法选择不捕捉可查异常时,它必须声明将抛出异常。

能够捕捉异常的方法,需要提供相符类型的异常处理器。所捕捉的异常,可能是由于自身语句所引发并抛出的异常,也可能是由某个调用的方法或者 Java 运行时 系统等抛出的异常。也就是说,一个方法所能捕捉的异常,一定是 Java 代码在某处所抛出的异常。简单地说,异常总是先被抛出,后被捕捉的。

任何 Java 代码都可以抛出异常,如:自己编写的代码、来自 Java 开发环境包中代码,或者 Java 运行时系统。无论是谁,都可以通过 Java 的 throw 语句抛出异常。

从方法中抛出的任何异常都必须使用 throws 子句。

捕捉异常通过 try-catch 语句或者 try-catch-finally 语句实现。

总体来说,Java 规定:对于可查异常必须捕捉、或者声明抛出。允许忽略不可查的 RuntimeException 和 Error。

4.1 捕获异常:try、catch 和 finally

1.try-catch 语句

在 Java 中,异常通过 try-catch 语句捕获。其一般语法形式为:

  1. try {
  2.     // 可能会发生异常的程序代码  
  3. catch (Type1 id1){
  4.     // 捕获并处置 try 抛出的异常类型 Type1  
  5. }
  6. catch (Type2 id2){
  7.      //捕获并处置 try 抛出的异常类型 Type2  
  8. }

关键词 try 后的一对大括号将一块可能发生异常的代码包起来,称为监控区域。Java 方法在运行过程中出现异常,则创建异常对象。将异常抛出监控区域之 外,由 Java 运行时系统试图寻找匹配的 catch 子句以捕获异常。若有匹配的 catch 子句,则运行其异常处理代码,try-catch 语句结束。

匹配的原则是:如果抛出的异常对象属于 catch 子句的异常类,或者属于该异常类的子类,则认为生成的异常对象与 catch 块捕获的异常类型相匹配。

例 1  捕捉 throw 语句抛出的“除数为 0”异常。

  1. public class TestException {
  2.     public static void main(String[] args) {
  3.         int a = 6;
  4.         int b = 0;
  5.         try { // try 监控区域  
  6.             if (b == 0throw new ArithmeticException(); // 通过 throw 语句抛出异常  
  7.             System.out.println(“a/b 的值是:” + a / b);
  8.         }
  9.         catch (ArithmeticException e) { // catch 捕捉异常  
  10.             System.out.println(“程序出现异常,变量 b 不能为 0。”);
  11.         }
  12.         System.out.println(“程序正常结束。”);
  13.     }
  14. }

运行结果:

程序出现异常,变量 b 不能为 0。

程序正常结束。

例 1  在 try 监控区域通过 if 语句进行判断,当“除数为 0”的错误条件成立时引发 ArithmeticException 异常,创建 ArithmeticException 异常对象,并由 throw 语句将异常抛给 Java 运行时系统,由系统寻找匹配的异常处理器 catch 并运行相应异 常处理代码,打印输出“程序出现异常,变量 b 不能为 0。”try-catch 语句结束,继续程序流程。

事实上,“除数为 0”等 ArithmeticException,是 RuntimException 的子类。而运行时异常将由运行时系统自动抛出,不需要使用 throw 语句。

例 2  捕捉运行时系统自动抛出“除数为 0”引发的 ArithmeticException 异常。

  1.  public static void main(String[] args) {
  2.         int a = 6;
  3.         int b = 0;
  4.         try {
  5.             System.out.println(“a/b 的值是:” + a / b);
  6.         } catch (ArithmeticException e) {
  7.             System.out.println(“程序出现异常,变量 b 不能为 0。”);
  8.         }
  9.         System.out.println(“程序正常结束。”);
  10.     }
  11. }

运行结果:程序出现异常,变量 b 不能为 0。

程序正常结束。

例 2  中的语句:

System.out.println(“a/b 的值是:” + a/b);

在运行中出现“除数为 0”错误,引发 ArithmeticException 异常。运行时系统创建异常对象并抛出监控区域,转而匹配合适的异常处理器 catch,并执行相应的异常处理代码。

由于检查运行时异常的代价远大于捕捉异常所带来的益处,运行时异常不可查。Java 编译器允许忽略运行时异常,一个方法可以既不捕捉,也不声明抛出运行时异常。

例 3  不捕捉、也不声明抛出运行时异常。

  1. public class TestException {
  2.     public static void main(String[] args) {
  3.         int a, b;
  4.         a = 6;
  5.         b = 0// 除数 b 的值为 0  
  6.         System.out.println(a / b);
  7.     }
  8. }

运行结果:

Exception in thread “main” java.lang.ArithmeticException: / by zero
at Test.TestException.main(TestException.java:8)

例 4  程序可能存在除数为 0 异常和数组下标越界异常。

  1. public class TestException {
  2.     public static void main(String[] args) {
  3.         int[] intArray = new int[3];
  4.         try {
  5.             for (int i = 0; i <= intArray.length; i++) {
  6.                 intArray[i] = i;
  7.                 System.out.println(“intArray[“ + i + “] = “ + intArray[i]);
  8.                 System.out.println(“intArray[“ + i + “]模 “ + (i – 2) + “的值:  “
  9.                         + intArray[i] % (i – 2));
  10.             }
  11.         } catch (ArrayIndexOutOfBoundsException e) {
  12.             System.out.println(“intArray 数组下标越界异常。”);
  13.         } catch (ArithmeticException e) {
  14.             System.out.println(“除数为 0 异常。”);
  15.         }
  16.         System.out.println(“程序正常结束。”);
  17.     }
  18. }

运行结果:

intArray[0] = 0

intArray[0]模 -2 的值:  0

intArray[1] = 1

intArray[1]模 -1 的值:  0

intArray[2] = 2

除数为 0 异常。

程序正常结束。

例 4  程序可能会出现除数为 0 异常,还可能会出现数组下标越界异常。程序运行过程中 ArithmeticException 异常类型是先行匹配的,因此执行相匹配的 catch 语句:

  1. catch (ArithmeticException e){
  2.       System.out.println(“除数为 0 异常。”);
  3.  }

需要注意的是,一旦某个 catch 捕获到匹配的异常类型,将进入异常处理代码。一经处理结束,就意味着整个 try-catch 语句结束。其他的 catch 子句不再有匹配和捕获异常类型的机会。

Java 通过异常类描述异常类型,异常类的层次结构如图 1 所示。对于有多个 catch 子句的异常程序而言,应该尽量将捕获底层异常类的 catch 子 句放在前面,同时尽量将捕获相对高层的异常类的 catch 子句放在后面。否则,捕获底层异常类的 catch 子句将可能会被屏蔽。

RuntimeException 异常类包括运行时各种常见的异常,ArithmeticException 类和 ArrayIndexOutOfBoundsException 类都是它的子类。因此,RuntimeException 异常类的 catch 子句应该放在 最后面,否则可能会屏蔽其后的特定异常处理或引起编译错误。

2. try-catch-finally 语句

try-catch 语句还可以包括第三部分,就是 finally 子句。它表示无论是否出现异常,都应当执行的内容。try-catch-finally 语句的一般语法形式为:

  1. try {
  2.        // 可能会发生异常的程序代码  
  3.    } catch (Type1 id1) {
  4.        // 捕获并处理 try 抛出的异常类型 Type1  
  5.    } catch (Type2 id2) {
  6.        // 捕获并处理 try 抛出的异常类型 Type2  
  7.    } finally {
  8.        // 无论是否发生异常,都将执行的语句块  
  9.    }

例 5  带 finally 子句的异常处理程序。

  1. public class TestException {
  2.     public static void main(String args[]) {
  3.         int i = 0;
  4.         String greetings[] = { ” Hello world !”” Hello World !! “,
  5.                 ” HELLO WORLD !!!” };
  6.         while (i < 4) {
  7.             try {
  8.                 // 特别注意循环控制变量 i 的设计,避免造成无限循环  
  9.                 System.out.println(greetings[i++]);
  10.             } catch (ArrayIndexOutOfBoundsException e) {
  11.                 System.out.println(“数组下标越界异常”);
  12.             } finally {
  13.                 System.out.println(“————————–“);
  14.             }
  15.         }
  16.     }
  17. }

运行结果:

Hello world !

————————–

Hello World !!

————————–

HELLO WORLD !!!

————————–

数组下标越界异常

————————–

在例 5 中,请特别注意 try 子句中语句块的设计,如果设计为如下,将会出现死循环。如果设计为:

  1. try {
  2.       System.out.println (greetings[i]); i++;
  3. }

 

小结:

try 块:用于捕获异常。其后可接零个或多个 catch 块,如果没有 catch 块,则必须跟一个 finally 块。

catch 块:用于处理 try 捕获到的异常。

finally 块:无论是否捕获或处理异常,finally 块里的语句都会被执行。当在 try 块或 catch 块中遇到 return 语句时,finally 语句块将在方法返回之前被执行。在以下 4 种特殊情况下,finally 块不会被执行:

1)在 finally 语句块中发生了异常。

2)在前面的代码中用了 System.exit()退出程序。

3)程序所在的线程死亡。

4)关闭 CPU。

3. try-catch-finally 规则(异常处理语句的语法规则):

1) 必须在 try 之后添加 catch 或 finally 块。try 块后可同时接 catch 和 finally 块,但至少有一个块。
2) 必须遵循块顺序:若代码同时使用 catch 和 finally 块,则必须将 catch 块放在 try 块之后。
3) catch 块与相应的异常类的类型相关。
4) 一个 try 块可能有多个 catch 块。若如此,则执行第一个匹配块。即 Java 虚拟机会把实际抛出的异常对象依次和各个 catch 代码块声明的异常类型匹配,如果异常对象为某个异常类型或其子类的实例,就执行这个 catch 代码块,不会再执行其他的 catch 代码块
5) 可嵌套 try-catch-finally 结构。
6) 在 try-catch-finally 结构中,可重新抛出异常。
7) 除了下列情况,总将执行 finally 做为结束:JVM 过早终止(调用 System.exit(int));在 finally 块中抛出一个未处理的异常;计算机断电、失火、或遭遇病毒攻击。

4. try、catch、finally 语句块的执行顺序:

1)当 try 没有捕获到异常时:try 语句块中的语句逐一被执行,程序将跳过 catch 语句块,执行 finally 语句块和其后的语句;

2)当 try 捕获到异常,catch 语句块里没有处理此异常的情况:当 try 语句块里的某条语句出现异常时,而没有处理此异常的 catch 语句块时,此异常将会抛给 JVM 处理,finally 语句块里的语句还是会被执行,但 finally 语句块后的语句不会被执行;

3)当 try 捕获到异常,catch 语句块里有处理此异常的情况:在 try 语句块中是按照顺序来执行的,当执行到某一条语句出现异常时,程序将跳到 catch 语句块,并与 catch 语句块逐一匹配,找到与之对应的处理程序,其他的 catch 语句块将不会被执行,而 try 语句块中,出现异常之后的语句也不会被执行,catch 语句块执行完后,执行 finally 语句块里的语句,最后执行 finally 语句块后的语句;
图示 try、catch、finally 语句块的执行:

深入理解 Java 异常处理机制

图 2  图示 try、catch、finally 语句块的执行

4.2 抛出异常

      任何 Java 代码都可以抛出异常,如:自己编写的代码、来自 Java 开发环境包中代码,或者 Java 运行时系统。无论是谁,都可以通过 Java 的 throw 语句抛出异常。从方法中抛出的任何异常都必须使用 throws 子句。

1. throws 抛出异常

如果一个方法可能会出现异常,但没有能力处理这种异常,可以在方法声明处用 throws 子句来声明抛出异常。例如汽车在运行时可能会出现故障,汽车本身没办法处理这个故障,那就让开车的人来处理。

throws 语句用在方法定义时声明该方法要抛出的异常类型,如果抛出的是 Exception 异常类型,则该方法被声明为抛出所有的异常。多个异常可使用逗号分割。throws 语句的语法格式为:

  1. methodname throws Exception1,Exception2,..,ExceptionN
  2. {
  3. }

方法名后的 throws Exception1,Exception2,…,ExceptionN 为声明要抛出的异常列表。当方法抛出异常列表的异常时,方法将不对这些类型及其子类类型的异常作处理,而抛向调用该方法的方法,由他去处理。例如:

  1. import java.lang.Exception;
  2. public class TestException {
  3.     static void pop() throws NegativeArraySizeException {
  4.         // 定义方法并抛出 NegativeArraySizeException 异常  
  5.         int[] arr = new int[-3]; // 创建数组  
  6.     }
  7.     public static void main(String[] args) { // 主方法  
  8.         try { // try 语句处理异常信息  
  9.             pop(); // 调用 pop()方法  
  10.         } catch (NegativeArraySizeException e) {
  11.             System.out.println(“pop()方法抛出的异常”);// 输出异常信息  
  12.         }
  13.     }
  14. }

使用 throws 关键字将异常抛给调用者后,如果调用者不想处理该异常,可以继续向上抛出,但最终要有能够处理该异常的调用者。

pop 方法没有处理异常 NegativeArraySizeException,而是由 main 函数来处理。

Throws 抛出异常的规则:

1) 如果是不可查异常(unchecked exception),即 Error、RuntimeException 或它们的子类,那么可以不使用 throws 关键字来声明要抛出的异常,编译仍能顺利通过,但在运行时会被系统抛出。

2)必须声明方法可抛出的任何可查异常(checked exception)。即如果一个方法可能出现受可查异常,要么用 try-catch 语句捕获,要么用 throws 子句声明将它抛出,否则会导致编译错误

3)仅当抛出了异常,该方法的调用者才必须处理或者重新抛出该异常。当方法的调用者无力处理该异常的时候,应该继续抛出,而不是囫囵吞枣。

4)调用方法必须遵循任何可查异常的处理和声明规则。若覆盖一个方法,则不能声明与覆盖方法不同的异常。声明的任何异常必须是被覆盖方法所声明异常的同类或子类。

例如:

  1. void method1() throws IOException{}  //合法    
  2. //编译错误,必须捕获或声明抛出 IOException    
  3. void method2(){
  4.   method1();
  5. }
  6. //合法,声明抛出 IOException    
  7. void method3()throws IOException {
  8.   method1();
  9. }
  10. //合法,声明抛出 Exception,IOException 是 Exception 的子类    
  11. void method4()throws Exception {
  12.   method1();
  13. }
  14. //合法,捕获 IOException    
  15. void method5(){
  16.  try{
  17.     method1();
  18.  }catch(IOException e){…}
  19. }
  20. //编译错误,必须捕获或声明抛出 Exception    
  21. void method6(){
  22.   try{
  23.     method1();
  24.   }catch(IOException e){throw new Exception();}
  25. }
  26. //合法,声明抛出 Exception    
  27. void method7()throws Exception{
  28.  try{
  29.   method1();
  30.  }catch(IOException e){throw new Exception();}
  31. }

判断一个方法可能会出现异常的依据如下:
1)方法中有 throw 语句。例如,以上 method7()方法的 catch 代码块有 throw 语句。
2)调用了其他方法,其他方法用 throws 子句声明抛出某种异常。例如,method3()方法调用了 method1()方法,method1()方法声明抛出 IOException,因此,在 method3()方法中可能会出现 IOException。

2. 使用 throw 抛出异常

throw 总是出现在函数体中,用来抛出一个 Throwable 类型的异常。程序会在 throw 语句后立即终止,它后面的语句执行不到,然后在包含它的所有 try 块中(可能在上层调用函数中)从里向外寻找含有与其匹配的 catch 子句的 try 块。
我们知道,异常是异常类的实例对象,我们可以创建异常类的实例对象通过 throw 语句抛出。该语句的语法格式为:
throw new exceptionname;
例如抛出一个 IOException 类的异常对象:
throw new IOException;
要注意的是,throw 抛出的只能够是可抛出类 Throwable 或者其子类的实例对象。下面的操作是错误的:
throw new String(“exception”);

这是因为 String 不是 Throwable 类的子类。

如果抛出了检查异常,则还应该在方法头部声明方法可能抛出的异常类型。该方法的调用者也必须检查处理抛出的异常。

如果所有方法都层层上抛获取的异常,最终 JVM 会进行处理,处理也很简单,就是打印异常消息和堆栈信息。如果抛出的是 Error 或 RuntimeException,则该方法的调用者可选择处理该异常。

  1. package Test;
  2. import java.lang.Exception;
  3. public class TestException {
  4.     static int quotient(int x, int y) throws MyException { // 定义方法抛出异常  
  5.         if (y < 0) { // 判断参数是否小于 0  
  6.             throw new MyException(“除数不能是负数”); // 异常信息  
  7.         }
  8.         return x/y; // 返回值  
  9.     }
  10.     public static void main(String args[]) { // 主方法  
  11.         int  a =3;
  12.         int  b =0;
  13.         try { // try 语句包含可能发生异常的语句  
  14.             int result = quotient(a, b); // 调用方法 quotient()  
  15.         } catch (MyException e) { // 处理自定义异常  
  16.             System.out.println(e.getMessage()); // 输出异常信息  
  17.         } catch (ArithmeticException e) { // 处理 ArithmeticException 异常  
  18.             System.out.println(“除数不能为 0”); // 输出提示信息  
  19.         } catch (Exception e) { // 处理其他异常  
  20.             System.out.println(“程序发生了其他的异常”); // 输出提示信息  
  21.         }
  22.     }
  23. }
  24. class MyException extends Exception { // 创建自定义异常类  
  25.     String message; // 定义 String 类型变量  
  26.     public MyException(String ErrorMessagr) { // 父类方法  
  27.         message = ErrorMessagr;
  28.     }
  29.     public String getMessage() { // 覆盖 getMessage()方法  
  30.         return message;
  31.     }
  32. }

4.3 异常链

1) 如果调用 quotient(3,-1),将发生 MyException 异常,程序调转到 catch (MyException e)代码块中执行;

2) 如果调用 quotient(5,0),将会因“除数为 0”错误引发 ArithmeticException 异常,属于运行时异常类,由 Java 运行时系统自动抛出。quotient()方法没有捕捉 ArithmeticException 异常,Java 运行时系统将沿方法调用栈查到 main 方法,将抛出的异常上传至 quotient()方法的调用者:

int result = quotient(a, b); // 调用方法 quotient()
由于该语句在 try 监控区域内,因此传回的“除数为 0”的 ArithmeticException 异常由 Java 运行时系统抛出,并匹配 catch 子句:

catch (ArithmeticException e) { // 处理 ArithmeticException 异常
System.out.println(“除数不能为 0”); // 输出提示信息
}

处理结果是输出“除数不能为 0”。Java 这种向上传递异常信息的处理机制,形成异常链

Java 方法抛出的可查异常将依据调用栈、沿着方法调用的层次结构一直传递到具备处理能力的调用方法,最高层次到 main 方法为止。如果异常传递到 main 方法,而 main 不具备处理能力,也没有通过 throws 声明抛出该异常,将可能出现编译错误。

3)如还有其他异常发生,将使用 catch (Exception e)捕捉异常。由于 Exception 是所有异常类的父类,如果将 catch (Exception e)代码块放在其他两个代码块的前面,后面的代码块将永远得不到执行,就没有什么意义了,所以 catch 语句的顺序不可掉换。

4.4 Throwable 类中的常用方法

注意:catch 关键字后面括号中的 Exception 类型的参数 e。Exception 就是 try 代码块传递给 catch 代码块的变量类型,e 就是变量名。catch 代码块中语句”e.getMessage();”用于输出错误性质。通常异常处理常用 3 个函数来获取异常的有关信息:

getCause():返回抛出异常的原因。如果 cause 不存在或未知,则返回 null。

getMeage():返回异常的消息信息。

printStackTrace():对象的堆栈跟踪输出至错误输出流,作为字段 System.err 的值。

有时为了简单会忽略掉 catch 语句后的代码,这样 try-catch 语句就成了一种摆设,一旦程序在运行过程中出现了异常,就会忽略处理异常,而错误发生的原因很难查找。

 5.Java 常见异常

 

在 Java 中提供了一些异常用来描述经常发生的错误,对于这些异常,有的需要程序员进行捕获处理或声明抛出,有的是由 Java 虚拟机自动进行捕获处理。Java 中常见的异常类:

1. runtimeException 子类:

    1、 java.lang.ArrayIndexOutOfBoundsException
数组索引越界异常。当对数组的索引值为负数或大于等于数组大小时抛出。
2、java.lang.ArithmeticException
算术条件异常。譬如:整数除零等。
3、java.lang.NullPointerException
空指针异常。当应用试图在要求使用对象的地方使用了 null 时,抛出该异常。譬如:调用 null 对象的实例方法、访问 null 对象的属性、计算 null 对象的长度、使用 throw 语句抛出 null 等等
4、java.lang.ClassNotFoundException
找不到类异常。当应用试图根据字符串形式的类名构造类,而在遍历 CLASSPAH 之后找不到对应名称的 class 文件时,抛出该异常。   5、java.lang.NegativeArraySizeException  数组长度为负异常

6、java.lang.ArrayStoreException 数组中包含不兼容的值抛出的异常

7、java.lang.SecurityException 安全性异常

8、java.lang.IllegalArgumentException 非法参数异常

2.IOException

IOException:操作输入流和输出流时可能出现的异常。

EOFException   文件已结束异常

FileNotFoundException   文件未找到异常

3. 其他

ClassCastException    类型转换异常类

ArrayStoreException  数组中包含不兼容的值抛出的异常

SQLException   操作数据库异常类

NoSuchFieldException   字段未找到异常

NoSuchMethodException   方法未找到抛出的异常

NumberFormatException    字符串转换为数字抛出的异常

StringIndexOutOfBoundsException 字符串索引超出范围抛出的异常

IllegalAccessException  不允许访问某类异常

InstantiationException  当应用程序试图使用 Class 类中的 newInstance()方法创建一个类的实例,而指定的类对象无法被实例化时,抛出该异常

 6.自定义异常

使用 Java 内置的异常类可以描述在编程时出现的大部分异常情况。除此之外,用户还可以自定义异常。用户自定义异常类,只需继承 Exception 类即可。
在程序中使用自定义异常类,大体可分为以下几个步骤。
(1)创建自定义异常类。
(2)在方法中通过 throw 关键字抛出异常对象。
(3)如果在当前抛出异常的方法中处理异常,可以使用 try-catch 语句捕获并处理;否则在方法的声明处通过 throws 关键字指明要抛出给方法调用者的异常,继续进行下一步操作。
(4)在出现异常方法的调用者中捕获并处理异常。

在上面的“使用 throw 抛出异常”例子已经提到了。

 

原文地址:http://blog.csdn.net/hguisu/article/details/6155636


何处钟 , 版权所有丨如未注明 , 均为原创丨本网站采用BY-NC-SA协议进行授权
转载请注明原文链接:深入理解 Java 异常处理机制
喜欢 (0)
[15211539367@163.com]
分享 (0)

您必须 登录 才能发表评论!