Java 语言要求 java 程序中(无论是谁写的代码),所有抛出( throw )的异常都必须是从 Throwable 派生而来。 当然,实际的 Java 编程中,由于 JDK 平台已经为我们设计好了非常丰富和完整的异常对象分类模型。因此, java 程序员一般是不需要再重新定义自己的异常对象。而且即便是需要扩展自定义的异常对象,也往往会从 Exception 派生而来。所以,对于 java 程序员而言,它一般只需要在它的顶级函数中 catch(Exception ex) 就可以捕获出所有的异常对象。 所有异常对象的根基类是 Throwable , Throwable 从 Object 直接继承而来(这是 java 系统所强制要求的),并且它实现了 Serializable 接口(这为所有的异常对象都能够轻松跨越 Java 组件系统做好了最充分的物质准备)。从 Throwable 直接派生出的异常类有 Exception 和 Error 。 Exception 是 java 程序员所最熟悉的,它一般代表了真正实际意义上的异常对象的根基类。也即是说, Exception 和从它派生而来的所有异常都是应用程序能够 catch 到的,并且可以进行异常错误恢复处理的异常类型。而 Error 则表示 Java 系统中出现了一个非常严重的异常错误,并且这个错误可能是应用程序所不能恢复的,例如 LinkageError ,或 ThreadDeath 等。
首先还是看一个例子吧!代码如下: import java.io.*; public class Trans { public static void main(String[] args) { try { BufferedReader rd=null; Writer wr=null; try { File srcFile = new File((args[0])); File dstFile = new File((args[1])); rd = new BufferedReader(new InputStreamReader(new FileInputStream(srcFile), args[2])); wr = new OutputStreamWriter(new FileOutputStream(dstFile), args[3]); // 注意下面这条语句,它有什么问题吗? if (rd == null || wr == null) throw new Exception("error! test!"); while(true) { String sLine = rd.readLine(); if(sLine == null) break; wr.write(sLine); wr.write("\r\n"); } } finally { wr.flush(); wr.close(); rd.close(); } } catch(IOException ex) { ex.printStackTrace(); } } } 熟悉 java 语言的程序员朋友们,你们认为上面的程序有什么问题吗?编译能通过吗?如果不能,那么原因又是为何呢?好了,有了自己的分析和预期之后,不妨亲自动手编译一下上面的小程序,呵呵!结果确实如您所料?是的,的确是编译时报错了,错误信息如下: E:\Trans.java:20: unreported exception java.lang.Exception; must be caught or declared to be thrown if (rd == null || wr == null) throw new Exception("error! test!"); 1 error 上面这种编译错误信息,相信 Java 程序员肯定见过(可能还是屡见不鲜!)! 相信老练一些的 Java 程序员一定非常清楚上述编译出错的原因。那就是如错误信息中(“ must be caught ”)描述的那样, 在 Java 的异常处理模型中,要求所有被抛出的异常都必须要有对应的“异常处理模块” 。也即是说,如果你在程序中 throw 出一个异常,那么在你的程序中(函数中)就必须要 catch 这个异常(处理这个异常)。例如上面的例子中,你在第 20 行代码处,抛出了一个 Exception 类型的异常,但是在该函数中,却没有 catch 并处理掉此异常的地方。因此,这样的程序即便是能够编译通过,那么运行时也是致命的(可能导致程序的崩溃),所以, Java 语言干脆在编译时就尽可能地检查(并卡住)这种本不应该出现的错误,这无疑对提高程序的可靠性大有帮助。 但是,在 Java 语言中,这就是必须的。 如果一个函数中,它运行时可能会向上层调用者函数抛出一个异常,那么,它就必须在该函数的声明中显式的注明(采用 throws 关键字) 。还记得刚才那条编译错误信息吗?“ must be caught or declared to be thrown ”,其中“ must be caught ”上面已经解释了,而后半部分呢?“ declared to be thrown ”是指何意呢?其实指的就是“必须显式地声明某个函数可能会向外部抛出一个异常”,也即是说,如果一个函数内部,它可能抛出了一种类型的异常,但该函数内部并不想(或不宜) catch 并处理这种类型的异常,此时,它就必须使用 throws 关键字来声明该函数可能会向外部抛出一个异常,以便于该函数的调用者知晓并能够及时处理这种类型的异常。下面列出了这几种情况的比较,代码如下: // 示例程序 1 ,这种写法能够编译通过 package com.ginger.exception; import java.io.*; public class Trans { public static void main(String[] args) { try { test(); } catch (Exception ex) { ex.printStackTrace(); } } static void test() { try { throw new Exception("To show Exception Successed"); } catch (Exception ex) { ex.printStackTrace(); } } } // 示例程序 2 ,这种写法就不能够编译通过 import java.io.*; public class Trans { public static void main(String[] args) { try { test(); } // 虽然这里能够捕获到 Exception 类型的异常 catch (Exception ex) { ex.printStackTrace(); } } static void test() { throw new Exception("test"); } } // 示例程序 3 ,这种写法又能够被编译通过 import java.io.*; public class Trans { public static void main(String[] args) { try { test(); } catch(Exception ex) { ex.printStackTrace(); } } // 由于函数声明了可能抛出 Exception 类型的异常 static void test() throws Exception { throw new Exception("test"); } } // 示例程序 4 ,它又不能够被编译通过了 import java.io.*; public class Trans { public static void main(String[] args) { try { // 虽然 test() 函数并没有真正抛出一个 Exception 类型的异常 // 但是由于它函数声明时,表示它可能抛出一个 Exception 类型的异常 // 所以,这里仍然不能被编译通过。 // 呵呵!体会到了 Java 异常处理模型的严谨吧! test(); } catch(IOException ex) { ex.printStackTrace(); } } static void test() throws Exception { } }
总结:用throw时,一定要在函数体内要有对应的catch(),意思是说,抛出IOException e,你就要catch IOException,抛出Exception,就要catch Exception;要对应。 用throws时,上层调用者去catch()就可以了.
不知上面几个有联系的示例是否能够给大家带来“豁然开朗”的感觉,坦率的说, Java 提供的异常处理模型并不复杂,相信太多太多 Java 程序员有着比我更深刻的认识。最后,补充一种例外情况,请看如下代码: import java.io.*; public class Trans { public static void main(String[] args) { try { test(); } catch (Exception ex) { ex.printStackTrace(); } } static void test() throws Error { throw new Error(" 故意抛出一个 Error"); } } 朋友们!上面的程序能被编译通过吗?注意,按照刚才上面所总结出的规律:在 Java 的异常处理模型中,要求所有被抛出的异常都必须要有对应的 catch 块!那么上面的程序肯定不能被编译通过,因为 Error 和 Exception 都是从 Throwable 直接派生而来,而 test 函数声明了它可能抛出 Error 类型的异常,但在 main 函数中却并没有 catch(Error) 或 catch(Throwable) 块,所以它理当是会编译出错的!真的吗?不妨试试!呵呵!结果并非我们之预料,而它恰恰是正确编译通过了。为何? WHY ? WHY ? 其实,原因很简单,那就是因为 Error 异常的特殊性。 Java 异常处理模型中规定: Error 和从它派生而来的所有异常,都表示系统中出现了一个非常严重的异常错误,并且这个错误可能是应用程序所不能恢复的 (其实这在前面的内容中已提到过)。因此,如果系统中真的出现了一个 Error 类型的异常,那么则表明,系统已处于崩溃不可恢复的状态中,此时,作为编写 Java 应用程序的你,已经是没有必要(也没能力)来处理此等异常错误。所以, javac 编译器就没有必要来保证:“在编译时,所有的 Error 异常都有其对应的错误处理模块”。当然, Error 类型的异常一般都是由系统遇到致命的错误时所抛出的,它最后也由 Java 虚拟机所处理。而作为 Java 程序员的你,可能永远也不会考虑抛出一个 Error 类型的异常。因此 Error 是一个特例情况! 特别关注一下 RuntimeException 上面刚刚讨论了一下 Error 类型的异常处理情况, Java 程序员一般无须关注它(处理这种异常)。另外,其实在 Exception 类型的异常对象中,也存在一种比较特别的“异常”类型,那就是 RuntimeException ,虽然它是直接从 Exception 派生而来,但是 Java 编译器( javac )对 RuntimeException 却是特殊待遇,而且是照顾有加。不信,看看下面的两个示例吧!代码如下: // 示例程序 1 // 它不能编译通过,我们可以理解 import java.io.*;
public class Trans { public static void main(String[] args) { test(); }
static void test() { // 注意这条语句 throw new Exception(" 故意抛出一个 Exception"); } } // 示例程序 2 // 可它却为什么能够编译通过呢? import java.io.*;
public class Trans { public static void main(String[] args) { test(); }
static void test() { // 注意这条语句 throw new RuntimeException(" 故意抛出一个 RuntimeException"); } } 对上面两个相当类似的程序, javac 编译时却遭遇了两种截然不同的处理,按理说,第 2 个示例程序也应该像第 1 个示例程序那样,编译时报错!但是 javac 编译它时,却例外地让它通过它,而且在运行时, java 虚拟机也捕获到了这个异常,并且会在 console 打印出详细的异常信息。运行结果如下: java.lang.RuntimeException: 故意抛出一个 RuntimeException at Trans.test(Trans.java:13) at Trans.main(Trans.java:8) Exception in thread "main" 为什么对于 RuntimeException 类型的异常(以及从它派生而出的异常类型), javac 和 java 虚拟机都特殊处理呢?要知道,这可是与“ Java 异常处理模型更严谨和更安全”的设计原则相抵触的呀!究竟是为何呢?这简直让人不法理解呀! 只不过, Java 语言中, RuntimeException 被统一纳入到了 Java 语言和 JDK 的规范之中。请看如下代码,来验证一下我们的理解! import java.io.*; public class Trans { public static void main(String[] args) { test(); } static void test() { int i = 4; int j = 0; // 运行时,这里将触发了一个 ArithmeticException // ArithmeticException 从 RuntimeException 派生而来 System.out.println("i / j = " + i / j); } } 运行结果如下: java.lang.ArithmeticException: / by zero at Trans.test(Trans.java:16) at Trans.main(Trans.java:8) Exception in thread "main" 又如下面的例子,也会产生一个 RuntimeException ,代码如下: import java.io.*; public class Trans { public static void main(String[] args) { test(); } static void test() { String str = null; // 运行时,这里将触发了一个 NullPointerException // NullPointerException 从 RuntimeException 派生而来 str.compareTo("abc"); } } 所以,针对 RuntimeException 类型的异常, javac 是无法通过编译时的静态语法检测来判断到底哪些函数(或哪些区域的代码)可能抛出这类异常(这完全取决于运行时状态,或者说运行态所决定的),也正因为如此, Java 异常处理模型中的“ must be caught or declared to be thrown ”规则也不适用于 RuntimeException (所以才有前面所提到过的奇怪编译现象,这也属于特殊规则吧)。但是, Java 虚拟机却需要有效地捕获并处理此类异常。当然, RuntimeException 也可以被程序员显式地抛出,而且为了程序的可靠性,对一些可能出现“运行时异常( RuntimeException )”的代码区域,程序员最好能够及时地处理这些意外的异常,也即通过 catch(RuntimeExcetion) 或 catch(Exception) 来捕获它们。如下面的示例程序,代码如下: import java.io.*; public class Trans { public static void main(String[] args) { try { test(); } // 在上层的调用函数中,最好捕获所有的 Exception 异常! catch(Exception e) { System.out.println("go here!"); e.printStackTrace(); } } // 这里最好显式地声明一下,表明该函数可能抛出 RuntimeException static void test() throws RuntimeException { String str = null; // 运行时,这里将触发了一个 NullPointerException // NullPointerException 从 RuntimeException 派生而来 str.compareTo("abc"); } }
|
相关推荐
14.java异常处理机制.zip14.java异常处理机制.zip14.java异常处理机制.zip14.java异常处理机制.zip14.java异常处理机制.zip14.java异常处理机制.zip14.java异常处理机制.zip14.java异常处理机制.zip14.java异常处理...
本资源主要描述java异常的知识,里面内容丰富,是java初学者的学习手册,java异常机制小结。
描述java内存回收机制,异常出现原因,解决方案
深入理解java异常处理机制,很详细的,去了,你们的!
主要介绍了java异常处理机制示例(java抛出异常、捕获、断言),需要的朋友可以参考下
Java异常机制.txt
异常处理是Java语言的重要机制,正确、合理地处理异常对系统的健壮性和稳定性提供了强有力的支持。异常的处理主要包括捕捉异常、程序流程的跳转和异常处理语句块的定义等。
Java异常架构与异常关键字 Java异常简介 Java异常架构 1. Throwable 2. Error(错误) 3. Exception(异常) 运行时异常 编译时异常 4. 受检异常与非受检异常 受检异常 非受检异常 Java异常关键字 Java异常处理 声明...
java异常处理机制,异常的概念,发生的原因,throwable,捕获异常的简单思维导图
通过实现本文提出的Java异常机制的静态编译算法,Open64编译器可以正确编译Java程序。 栈展开是被编译器和Java虚拟机普遍使用的一种实现异常处理机制的方法。本文针对传统的栈展开算法中存在的不足,提出了一种改进的...
java异常练习题,北大青鸟学士后课后练习,主要是异常处理的小测验
JAVA的异常处理机制 java学习,值得学习
java 异常机制
对java异常的总结,详细描述了JAVA异常的分类,异常的抛出和捕捉的机制,以及在项目应用中应该如何对异常作单独处理
用户自行定义一个异常,编程创建并抛出某个异常类的实例,运行程序并观察执行结果 例如:用户密码的合法化验证,要求密码由4到六个数字组成,若长度不落在这个范围或不是由数字组成,抛出自己的异常
本文旨在探讨Java的\"异常机制\",分别介绍了\"异常类\"的组织形式、\"异常\"的处理过程、\"异常\"的处理方法及使用\"异常机制\"的需要注意的问题。
基于Java异常处理机制的研究,张军芳,肖华山,异常处理是Java语言的重要机制,有效地处理异常对程序的可靠性、健壮性是十分重要的。本文分析了异常处理机制的概念和指导原则,��