异常是程序中的一些错误,但并非全部的错误都是异常,而且错误有时候是能够避免的。java
好比说,你的代码少了一个分号,那么运行出来结果是提示是错误java.lang.Error
;若是你用System.out.println(11/0)
,那么你是由于你用0作了除数,会抛出java.lang.ArithmeticException
的异常。程序员
异常发生的缘由有不少,一般包含如下几大类:面试
这些异常有的是由于用户错误引发,有的是程序错误引发的,还有其它一些是由于物理错误引发的。-数据库
要理解Java异常处理是如何工做的,你须要掌握如下三种类型的异常:编程
异常指不期而至的各类情况,如:文件找不到、网络链接失败、除0操做、非法参数等。异常是一个事件,它发生在程序运行期间,干扰了正常的指令流程。数组
Java语言在设计的当初就考虑到这些问题,提出异常处理的框架的方案,全部的异常均可以用一个异常类来表示,不一样类型的异常对应不一样的子类异常(目前咱们所说的异常包括错误概念),定义异常处理的规范,在JDK1.4
版本之后增长了异常链机制,从而便于跟踪异常。安全
Java异常是一个描述在代码段中发生异常的对象,当发生异常状况时,一个表明该异常的对象被建立而且在致使该异常的方法中被抛出,而该方法能够选择本身处理异常或者传递该异常。网络
Java把异常看成对象来处理,并定义一个基类java.lang.Throwable
做为全部异常的超类。框架
在Java API中已经定义了许多异常类,这些异常类分为两大类,错误Error
和异常Exception
。ide
Java异常层次结构图以下图所示:
从图中能够看出全部异常类型都是内置类Throwable
的子类,于是Throwable
在异常类的层次结构的顶层。
接下来Throwable
分红了两个不一样的分支,一个分支是Error
,它表示不但愿被程序捕获或者是程序没法处理的错误。另外一个分支是Exception
,它表示用户程序可能捕捉的异常状况或者说是程序能够处理的异常。其中异常类Exception
又分为运行时异常(RuntimeException
)和非运行时异常。
Java异常又能够分为不受检查异常(Unchecked Exception
)和检查异常(Checked Exception
)。
下面将详细讲述这些异常之间的区别与联系:
Error
:Error
类对象由 Java 虚拟机生成并抛出,大多数错误与代码编写者所执行的操做无关。例如,Java虚拟机运行错误(Virtual MachineError
),当JVM再也不有继续执行操做所需的内存资源时,将出现 OutOfMemoryError
。这些异常发生时,Java虚拟机(JVM)通常会选择线程终止;还有发生在虚拟机试图执行应用时,如类定义错误(NoClassDefFoundError
)、连接错误(LinkageError
)。这些错误是不可查的,由于它们在应用程序的控制和处理能力之 外,并且绝大多数是程序运行时不容许出现的情况。对于设计合理的应用程序来讲,即便确实发生了错误,本质上也不该该试图去处理它所引发的异常情况。在Java中,错误一般是使用Error
的子类描述。Exception
:在Exception
分支中有一个重要的子类RuntimeException
(运行时异常),该类型的异常自动为你所编写的程序定义ArrayIndexOutOfBoundsException
(数组下标越界)、NullPointerException
(空指针异常)、ArithmeticException
(算术异常)、MissingResourceException
(丢失资源)、ClassNotFoundException
(找不到类)等异常,这些异常是不检查异常,程序中能够选择捕获处理,也能够不处理。这些异常通常是由程序逻辑错误引发的,程序应该从逻辑角度尽量避免这类异常的发生;而RuntimeException
以外的异常咱们统称为非运行时异常,类型上属于Exception
类及其子类,从程序语法角度讲是必须进行处理的异常,若是不处理,程序就不能编译经过。如IOException
、SQLException
等以及用户自定义的Exception
异常,通常状况下不自定义检查异常。
注意
Error
和Exception
的区别:Error
一般是灾难性的致命的错误,是程序没法控制和处理的,当出现这些异常时,Java虚拟机(JVM)通常会选择终止线程;Exception
一般状况下是能够被程序处理的,而且在程序中应该尽量的去处理这些异常。
检查异常
:在正确的程序运行过程当中,很容易出现的、情理可容的异常情况,在必定程度上这种异常的发生是能够预测的,而且一旦发生该种异常,就必须采起某种方式进行处理。♠提示
除了
RuntimeException
及其子类之外,其余的Exception
类及其子类都属于检查异常,当程序中可能出现这类异常,要么使用try-catch
语句进行捕获,要么用throws
子句抛出,不然编译没法经过。
不受检查异常
:包括RuntimeException
及其子类和Error
。♠提示
不受检查异常
为编译器不要求强制处理的异常,检查异常
则是编译器要求必须处置的异常。
Java的异常处理本质上是抛出异常和捕获异常。
抛出异常
:要理解抛出异常,首先要明白什么是异常情形(exception condition),它是指阻止当前方法或做用域继续执行的问题。其次把异常情形和普通问题相区分,普通问题是指在当前环境下能获得足够的信息,总能处理这个错误。对于异常情形,已经没法继续下去了,由于在当前环境下没法得到必要的信息来解决问题,你所能作的就是从当前环境中跳出,并把问题提交给上一级环境,这就是抛出异常时所发生的事情。抛出异常后,会有几件事随之发生。首先,是像建立普通的java对象同样将使用new
在堆上建立一个异常对象;而后,当前的执行路径(已经没法继续下去了)被终止,而且从当前环境中弹出对异常对象的引用。此时,异常处理机制接管程序,并开始寻找一个恰当的地方继续执行程序,这个恰当的地方就是异常处理程序或者异常处理器,它的任务是将程序从错误状态中恢复,以使程序要么换一种方式运行,要么继续运行下去。举个简单的例子,假使咱们建立了一个学生对象Student的一个引用stu,在调用的时候可能尚未初始化。因此在使用这个对象引用调用其余方法以前,要先对它进行检查,能够建立一个表明错误信息的对象,而且将它从当前环境中抛出,这样就把错误信息传播到更大的环境中。
if(stu == null){ throw new NullPointerException(); }
这就抛出了异常,它将在其余的地方获得执行或者处理,具体是哪一个地方后面将很快介绍,代码中出现的 throw
是一个关键字,暂时先不作过多讲解,后面会详细讲解。
捕获异常
:在方法抛出异常以后,运行时系统将转为寻找合适的异常处理器(exception handler)。潜在的异常处理器是异常发生时依次存留在调用栈中的方法的集合。当异常处理器所能处理的异常类型与方法抛出的异常类型相符时,即为合适的异常处理器。运行时系统从发生异常的方法开始,依次回查调用栈中的方法,直至找到含有合适异常处理器的方法并执行。当运行时系统遍历调用栈而未找到合适的异常处理器,则运行时系统终止。同时,意味着Java程序的终止。提示
对于
运行时异常
、错误
和检查异常
,Java技术所要求的异常处理方式有所不一样。因为运行时异常及其子类的不可查性,为了更合理、更容易地实现应用程序,Java规定,运行时异常将由Java运行时系统自动抛出,容许应用程序忽略运行时异常。
对于方法运行中可能出现的
Error
,当运行方法不欲捕捉时,Java容许该方法不作任何抛出声明。由于,大多数Error
异常属于永远不能被容许发生的情况,也属于合理的应用程序不应捕捉的异常。对于全部的检查异常,Java规定:一个方法必须捕捉,或者声明抛出方法以外。也就是说,当一个方法选择不捕捉检查异常时,它必须声明将抛出异常。
Java异常处理涉及到五个关键字,分别是:try
、catch
、finally
、throw
、throws
。下面将骤一介绍,经过认识这五个关键字,掌握基本异常处理知识。
• try -- 用于监听。将要被监听的代码(可能抛出异常的代码)放在try语句块以内,当try语句块内发生异常时,异常就被抛出。
• catch -- 用于捕获异常。catch用来捕获try语句块中发生的异常。
• finally -- finally语句块老是会被执行。它主要用于回收在try块里打开的物力资源(如数据库链接、网络链接和磁盘文件)。只有finally块,执行完成以后,才会回来执行try或者catch块中的return或者throw语句,若是finally中使用了return或者throw等终止方法的语句,则就不会跳回执行,直接中止。
• throw -- 用于抛出异常。
• throws -- 用在方法签名中,用于声明该方法可能抛出的异常。
try{ //code that might generate exceptions }catch(Exception e){ //the code of handling exception1 }catch(Exception e){ //the code of handling exception2 }
要明白异常捕获,还要理解监控区域
(guarded region)的概念。它是一段可能产生异常的代码,而且后面跟着处理这些异常的代码。
由此可见,上述try-catch
所描述的便是监控区域,关键词try
后的一对大括号将一块可能发生异常的代码包起来,即为监控区域。Java方法在运行过程当中发生了异常,则建立异常对象。将异常抛出监控区域以外,由Java运行时系统负责寻找匹配的catch
子句来捕获异常。如有一个catch
语句匹配到了,则执行该catch
块中的异常处理代码,就再也不尝试匹配别的catch
块了。
匹配的原则是:若是抛出的异常对象属于
catch
子句的异常类,或者属于该异常类的子类,则认为生成的异常对象与catch
块捕获的异常类型相匹配。
举个例子算术异常:
public class TestException { public static void main(String[] args) { int a = 1; int b = 0; try { // try监控区域 if (b == 0) throw new ArithmeticException(); // 经过throw语句抛出异常 System.out.println("a/b的值是:" + a / b); System.out.println("this will not be printed!"); } catch (ArithmeticException e) { // catch捕捉异常 System.out.println("程序出现异常,变量b不能为0!"); } System.out.println("程序正常结束。"); } }
运行结果:
D:\java>java TestException
程序出现异常,变量b不能为0!
程序正常结束。
显示一个异常的描述,
Throwable
重载了toString()
方法(由Object
定义),因此它将返回一个包含异常描述的字符串。例如,将前面的catch
块重写成:catch (ArithmeticException e) { // catch捕捉异常 System.out.println("程序出现异常"+e); }结果:
D:\java>java TestException 程序出现异常java.lang.ArithmeticException 程序正常结束。
根据前面讲述的,算术异常属于运行时异常,于是实际上该异常不须要程序抛出,运行时系统自动抛出,将例子改成以下:
public class TestException { public static void main(String[] args) { int a = 1; int b = 0; System.out.println("a/b的值是:" + a / b); System.out.println("this will not be printed!"); } }
结果:
D:\java>java TestException Exception in thread "main" java.lang.ArithmeticException: / by zero at TestException.main(TestException.java:7)
使用多重的catch语句
:不少状况下,由单个的代码段可能引发多个异常。处理这种状况,咱们须要定义两个或者更多的catch
子句,每一个子句捕获一种类型的异常,当异常被引起时,每一个catch
子句被依次检查,第一个匹配异常类型的子句执行,当一个catch
子句执行之后,其余的子句将被旁路。
编写多重catch语句块注意事项:
顺序问题:先小后大,即先子类后父类
Java经过异常类描述异常类型。对于有多个
catch
子句的异常程序而言,应该尽可能将捕获底层异常类的catch
子句放在前面,同时尽可能将捕获相对高层的异常类的catch
子句放在后面。不然,捕获底层异常类的catch
子句将可能会被屏蔽。
RuntimeException
异常类包括运行时各类常见的异常,ArithmeticException
类和ArrayIndexOutOfBoundsException
类都是它的子类。所以,RuntimeException
异常类的catch
子句应该放在最后面,不然可能会屏蔽其后的特定异常处理或引发编译错误。
嵌套try语句
:try
语句能够被嵌套。也就是说,一个try
语句能够在另外一个try
块的内部。每次进入try
语句,异常的先后关系都会被推入堆栈。若是一个内部的try
语句不含特殊异常的catch
处理程序,堆栈将弹出,下一个try
语句的catch
处理程序将检查是否与之匹配。这个过程将继续直到一个catch
语句被匹配成功,或者是直到全部的嵌套try
语句被检查完毕。若是没有catch
语句匹配,Java运行时系统将处理这个异常。
例如:
class NestTry{ public static void main(String[] args){ try{ int a = args.length; int b = 42 / a; System.out.println("a = "+ a); try{ if(a == 1){ a = a/(a-a); } if(a == 2){ int c[] = {1}; c[42] =99; } }catch(ArrayIndexOutOfBoundsException e){ System.out.println("ArrayIndexOutOfBounds :"+e); } }catch(ArithmeticException e){ System.out.println("Divide by 0"+ e); } } }
正如程序中所显示的,该程序在一个try块中嵌套了另外一个try
块。程序工做以下:当你在没有命令行参数的状况下执行该程序,外面的try
块将产生一个被0除的异常。程序在有一个命令行参数条件下执行,由嵌套的try
块产生一个被0除的异常,因为内部的catch
块不匹配这个异常,它将把异常传给外部的try
块,在外部异常被处理。若是你在具备两个命令行参数的条件下执行该程序,将由内部try
块产生一个数组边界异常。
结果:
D:\java>javac estTry.java D:\java>>java NestTry Divide by 0 java.lang.ArithmeticExceptio: / by zero D:\java>java NestTry one a = 1 Divide by 0java.lang.ArithmeticException: / by zero D:\java>java NestTry one two a = 2 ArrayIndexOutOfBounds :java.lang.ArrayIndexOutOfBoundsException: 42
注意:当有方法调用时,try
语句的嵌套能够很隐蔽的发生。例如,咱们能够将对方法的调用放在一个try
块中。在该方法的内部,有另外一个try
语句。在这种状况下,方法内部的try
仍然是嵌套在外部调用该方法的try
块中的。下面咱们将对上述例子进行修改,嵌套的try
块移到方法nesttry()的内部:
class NestTry{ static void nesttry(int a){ try{ if(a == 1){ a = a/(a-a); } if(a == 2){ int c[] = {1}; c[42] =99; } }catch(ArrayIndexOutOfBoundsException e){ System.out.println("ArrayIndexOutOfBounds :"+e); } } public static void main(String[] args){ try{ int a = args.length; int b = 42 / a; System.out.println("a = "+ a); nesttry(a); }catch(ArithmeticException e){ System.out.println("Divide by 0"+ e); } } }
结果输出与前面例子一致:
D:\java>javac NestTry.java D:\java>java NestTry Divide by 0java.lang.ArithmeticException: / by zero D:\java>java NestTry one a = 1 Divide by 0java.lang.ArithmeticException: / by zero D:\java>java NestTry one two a = 2 ArrayIndexOutOfBounds :java.lang.ArrayIndexOutOfBoundsException: 42
到目前为止,咱们只是获取了被Java运行时系统引起的异常。然而,咱们还能够用throw
语句抛出明确的异常。Throw
的语法形式以下:
throw ThrowableInstance;
这里的ThrowableInstance必定是Throwable
类类型或者Throwable
子类类型的一个对象。简单的数据类型,例如int
,char
,以及非Throwable
类,例如String
或Object
,不能用做异常。有两种方法能够获取Throwable
对象:在catch
子句中使用参数或者使用new
操做符建立。
程序执行完throw
语句以后当即中止;throw
后面的任何语句不被执行,最邻近的try
块用来检查它是否含有一个与异常类型匹配的catch
语句。若是发现了匹配的块,控制转向该语句;若是没有发现,次包围的try
块来检查,以此类推。若是没有发现匹配的catch
块,默认异常处理程序中断程序的执行而且打印堆栈轨迹。
例如:
class TestThrow{ static void proc(){ try{ throw new NullPointerException("demo"); }catch(NullPointerException e){ System.out.println("Caught inside proc"); throw e; } } public static void main(String [] args){ try{ proc(); }catch(NullPointerException e){ System.out.println("Recaught: "+e); } } }
结果:
D:\java>java TestThrow
Caught inside proc
Recaught: java.lang.NullPointerException: demo
该程序两次处理相同的错误,首先,main()
方法设立了一个异常关系而后调用proc()。proc()方法设立了另外一个异常处理关系而且当即抛出一个NullPointerException
实例,NullPointerException
在main()
中被再次捕获。
该程序阐述了怎样建立Java的标准异常对象,特别注意这一行:
throw new NullPointerException("demo");
此处
new
用来构造一个NullPointerException
实例,全部的Java内置的运行时异常有两个构造方法:一个没有参数,一个带有一个字符串参数。当用第二种形式时,参数指定描述异常的字符串。若是对象用做print()
或者println()
的参数时,该字符串被显示。这一样能够经过调用getMessage()来实现,getMessage()是由Throwable
定义的。
若是一个方法能够致使一个异常但不处理它,它必须指定这种行为以使方法的调用者能够保护它们本身而不发生异常。要作到这点,咱们能够在方法声明中包含一个throws
子句。一个throws
子句列举了一个方法可能引起的全部异常类型。这对于除了Error
或RuntimeException
及它们子类之外类型的全部异常是必要的。一个方法能够引起的全部其余类型的异常必须在throws
子句中声明,不然会致使编译错误。
下面是throws
子句的方法声明的通用形式:
public void info() throws Exception { //body of method }
Exception 是该方法可能引起的全部的异常,也能够是异常列表,中间以逗号隔开。
例如:
class TestThrows{ static void throw1(){ System.out.println("Inside throw1 . "); throw new IllegalAccessException("demo"); } public static void main(String[] args){ throw1(); } }
上述例子中有两个地方存在错误,你能看出来吗?
该例子中存在两个错误,首先,throw1()方法不想处理所致使的异常,于是它必须声明throws
子句来列举可能引起的异常即IllegalAccessException
;其次,main()
方法必须定义try/catch
语句来捕获该异常。
正确例子以下:
class TestThrows{ static void throw1() throws IllegalAccessException { System.out.println("Inside throw1 . "); throw new IllegalAccessException("demo"); } public static void main(String[] args){ try { throw1(); }catch(IllegalAccessException e ){ System.out.println("Caught " + e); } } }
Throws
抛出异常的规则:
- 若是是不受检查异常(
unchecked exception
),即Error
、RuntimeException
或它们的子类,那么能够不使用throws
关键字来声明要抛出的异常,编译仍能顺利经过,但在运行时会被系统抛出。- 必须声明方法可抛出的任何检查异常(
checked exception
)。即若是一个方法可能出现受可查异常,要么用try-catch
语句捕获,要么用throws
子句声明将它抛出,不然会致使编译错误- 仅当抛出了异常,该方法的调用者才必须处理或者从新抛出该异常。当方法的调用者无力处理该异常的时候,应该继续抛出,而不是囫囵吞枣。
- 调用方法必须遵循任何可查异常的处理和声明规则。若覆盖一个方法,则不能声明与覆盖方法不一样的异常。声明的任何异常必须是被覆盖方法所声明异常的同类或子类。
当异常发生时,一般方法的执行将作一个陡峭的非线性的转向,它甚至会过早的致使方法返回。例如,若是一个方法打开了一个文件并关闭,而后退出,你不但愿关闭文件的代码被异常处理机制旁路。finally
关键字为处理这种意外而设计。
finally
建立的代码块在try/catch
块完成以后另外一个try/catch
出现以前执行。finally
块不管有没有异常抛出都会执行。若是抛出异常,即便没有catch
子句匹配,finally
也会执行。一个方法将从一个try/catch
块返回到调用程序的任什么时候候,通过一个未捕获的异常或者是一个明确的返回语句,finally
子句在方法返回以前仍将执行。这在关闭文件句柄和释听任何在方法开始时被分配的其余资源是颇有用。
finally
子句是可选项,能够有也能够无,可是每一个try
语句至少须要一个catch
或者finally
子句。
class TestFinally{ static void proc1(){ try{ System.out.println("inside proc1"); throw new RuntimeException("demo"); }finally{ System.out.println("proc1's finally"); } } static void proc2(){ try{ System.out.println("inside proc2"); return ; } finally{ System.out.println("proc2's finally"); } } static void proc3(){ try{ System.out.println("inside proc3"); }finally{ System.out.println("proc3's finally"); } } public static void main(String [] args){ try{ proc1(); }catch(Exception e){ System.out.println("Exception caught"); } proc2(); proc3(); } }
该例子中,proc1()抛出了异常中断了try
,它的finally
子句在退出时执行。proc2的try
语句经过return
语句返回,但在返回以前finally
语句执行。在proc3()中try
语句正常执行,没有错误,finally
语句也被执行。
输出结果:
D:\java>java TestFinally inside proc1 proc1's finally Exception caught inside proc2 proc2's finally inside proc3 proc3's finally
注:若是
finally
块与一个try
联合使用,finally
块将在try
结束以前执行。
一、try{} 里有一个 return 语句,那么紧跟在这个 try 后的 finally{} 里的 code 会不会被执行,何时被执行,在 return 前仍是后?
答案:会执行,在方法返回调用者前执行。
注意:在finally中改变返回值的作法是很差的,由于若是存在finally代码块,try中的return语句不会立马返回调用者,而是记录下返回值待finally代码块执行完毕以后再向调用者返回其值,而后若是在finally中修改了返回值,就会返回修改后的值。显然,在finally中返回或者修改返回值会对程序形成很大的困扰,C#中直接用编译错误的方式来阻止程序员干这种龌龊的事情,Java中也能够经过提高编译器的语法检查级别来产生警告或错误,Eclipse中能够在如图所示的地方进行设置,强烈建议将此项设置为编译错误。
二、Java语言如何进行异常处理,关键字:throws、throw、try、catch、finally分别如何使用?
答:Java经过面向对象的方法进行异常处理,把各类不一样的异常进行分类,并提供了良好的接口。在Java中,每一个异常都是一个对象,它是Throwable类或其子类的实例。当一个方法出现异常后便抛出一个异常对象,该对象中包含有异常信息,调用这个对象的方法能够捕获到这个异常并能够对其进行处理。Java的异常处理是经过5个关键词来实现的:try、catch、throw、throws和finally。通常状况下是用try来执行一段程序,若是系统会抛出(throw)一个异常对象,能够经过它的类型来捕获(catch)它,或经过老是执行代码块(finally)来处理;try用来指定一块预防全部异常的程序;catch子句紧跟在try块后面,用来指定你想要捕获的异常的类型;throw语句用来明确地抛出一个异常;throws用来声明一个方法可能抛出的各类异常(固然声明异常时容许无病呻吟);finally为确保一段代码无论发生什么异常情况都要被执行;try语句能够嵌套,每当遇到一个try语句,异常的结构就会被放入异常栈中,直到全部的try语句都完成。若是下一级的try语句没有对某种异常进行处理,异常栈就会执行出栈操做,直到遇到有处理这种异常的try语句或者最终将异常抛给JVM。
三、运行时异常与受检异常有何异同?
答:异常表示程序运行过程当中可能出现的非正常状态,运行时异常表示虚拟机的一般操做中可能遇到的异常,是一种常见运行错误,只要程序设计得没有问题一般就不会发生。受检异常跟程序运行的上下文环境有关,即便程序设计无误,仍然可能因使用的问题而引起。Java编译器要求方法必须声明抛出可能发生的受检异常,可是并不要求必须声明抛出未被捕获的运行时异常。异常和继承同样,是面向对象程序设计中常常被滥用的东西,在Effective Java中对异常的使用给出了如下指导原则:
- 不要将异常处理用于正常的控制流(设计良好的API不该该强迫它的调用者为了正常的控制流而使用异常)
- 对能够恢复的状况使用受检异常,对编程错误使用运行时异常
- 避免没必要要的使用受检异常(能够经过一些状态检测手段来避免异常的发生)
- 优先使用标准的异常
- 每一个方法抛出的异常都要有文档
- 保持异常的原子性
- 不要在catch中忽略掉捕获到的异常
四、列出一些你常见的运行时异常?
答:
- ArithmeticException(算术异常)
- ClassCastException (类转换异常)
- IllegalArgumentException (非法参数异常)
- IndexOutOfBoundsException (下标越界异常)
- NullPointerException (空指针异常)
- SecurityException (安全异常)
异常链顾名思义就是将异常发生的缘由一个传一个串起来,即把底层的异常信息传给上层,这样逐层抛出。 Java API文档中给出了一个简单的模型:
try { lowLevelOp(); } catch (LowLevelException le) { throw (HighLevelException) new HighLevelException().initCause(le); }
当程序捕获到了一个底层异常,在处理部分选择了继续抛出一个更高级别的新异常给此方法的调用者。 这样异常的缘由就会逐层传递。这样,位于高层的异常递归调用getCause()方法,就能够遍历各层的异常缘由。 这就是Java异常链
的原理。异常链的实际应用不多,发生异常时候逐层上抛不是个好注意, 上层拿到这些异常又能奈之何?并且异常逐层上抛会消耗大量资源, 由于要保存一个完整的异常链信息.
使用Java内置的异常类能够描述在编程时出现的大部分异常状况。除此以外,用户还能够自定义异常。用户自定义异常类,只需继承Exception
类便可。
在程序中使用自定义异常类,大致可分为如下几个步骤:
throw
关键字抛出异常对象。try-catch
语句捕获并处理;不然在方法的声明处经过throws
关键字指明要抛出给方法调用者的异常,继续进行下一步操做。举例自定义异常:
class MyException extends Exception { private int detail; MyException(int a){ detail = a; } public String toString(){ return "MyException ["+ detail + "]"; } } public class TestMyException{ static void compute(int a) throws MyException{ System.out.println("Called compute(" + a + ")"); if(a > 10){ throw new MyException(a); } System.out.println("Normal exit!"); } public static void main(String [] args){ try{ compute(1); compute(20); }catch(MyException me){ System.out.println("Caught " + me); } } }
该例子彻底按照上述步骤。
运行结果以下: D:\java>java TestMyException Called compute(1) Normal exit! Called compute(20) Caught MyException [20]
参考文档: