有时候希望把刚捕获的异常重新抛出,尤其是在使用Exception捕获所有异常的时候。既然已经得到了对当前异常对象的引用,可以直接把它重新抛出:

catch(Exception e){
System.out.println("an exception was thrown");
throw e;
}

重新抛出异常会把异常抛给上一级环境中的异常处理程序,同一个try块的后续catch子句将会被忽略。此外,异常对象的所有信息都得以保持,所以高一级环境中捕获此异常的处理程序可以从这个异常对象中得到所有信息。

如果只是把当前异常对象重新抛出,那么printStackTrace()方法显示的将是原来异常抛出的调用栈信息,而非重新抛出点的信息。要向更新这个信息,可以调用filStackTrace()方法,这将返回一个Throwable对象,它是通过把当前调用栈信息填入原来那个异常对象而建立的,就像这样:

public class Rethrowing {

public static void f() throws Exception {
System.out.println("a_mark");
throw new Exception("b_mark");
} public static void g() throws Exception {
try {
f();
} catch (Exception e) {
System.out.println("c_mark");
e.printStackTrace(System.out);
throw e;
}
} public static void h() throws Exception {
try {
f();
} catch (Exception e) {
System.out.println("d_mark");
e.printStackTrace(System.out);
throw (Exception) e.fillInStackTrace();
}
} public static void main(String[] args) {
try {
g();
} catch (Exception e) {
System.out.println("e_mark");
e.printStackTrace(System.out);
} System.out.println("\n-------------between line--------------------\n"); try {
h();
} catch (Exception e) {
System.out.println("f_mark");
e.printStackTrace(System.out);
}
}

}

/*输出如下:

a_mark

c_mark

java.lang.Exception: b_mark

at testThread.Rethrowing.f(Rethrowing.java:7)

at testThread.Rethrowing.g(Rethrowing.java:12)

at testThread.Rethrowing.main(Rethrowing.java:32)

e_mark

java.lang.Exception: b_mark

at testThread.Rethrowing.f(Rethrowing.java:7)

at testThread.Rethrowing.g(Rethrowing.java:12)

at testThread.Rethrowing.main(Rethrowing.java:32)

-------------between line--------------------

a_mark

d_mark

java.lang.Exception: b_mark

at testThread.Rethrowing.f(Rethrowing.java:7)

at testThread.Rethrowing.h(Rethrowing.java:22)

at testThread.Rethrowing.main(Rethrowing.java:41)

f_mark

java.lang.Exception: b_mark

at testThread.Rethrowing.h(Rethrowing.java:26)

at testThread.Rethrowing.main(Rethrowing.java:41)

*/

调用filStackTrace()的那一行就成了异常的重新发生地了。

有可能在捕获异常之后抛出另一种异常。这么做的话,得到的效果类似于使用filStackTrace(),有关原来异常发生的信息会丢失,剩下的是与新的抛出点有关的信息:

class OneException extends Exception {

public OneException(String s) {

super(s);

}

}

class TwoException extends Exception {

public TwoException(String s) {

super(s);

}

}

public class RethrowNew {

public static void f() throws OneException {
System.out.println("a_mark");
throw new OneException("thrown from f()");
} public static void main(String[] args) {
try {
try {
f();
} catch (Exception e) {
System.out.println("b_mark");
e.printStackTrace(System.out);
throw new TwoException("from inner try");
}
} catch (Exception e) {
System.out.println("c_mark");
e.printStackTrace(System.out);
}
}

}

最后那个异常仅知道自己来自 main(),而对 f() 一无所知。

永远不必为清理前一个异常对象而担心,或者说为异常对象的清理而担心。他们都是用 new 在堆上创建的对象,所以gc会自动把他们清理掉。

备:throw 与 throws 之区分

throw 是语句抛出一个异常,该语句用在方法体内,表示抛出异常,由方法体内的语句处理。

throws 是方法可能抛出异常的声明,该语句用在方法声明后面,表示再抛出异常,由该方法的调用者来处理。

////end

java异常——重新抛出异常的更多相关文章

  1. Java 异常 重写抛出异常限制

    1 子类在重写父类抛出异常的方法时,要么不抛出异常,要么抛出与父类方法相同的异常或该异常的子类.如果被重写的父类方法只抛出受检异常,则子类重写的方法可以抛出非受检异常.例如,父类方法抛出了一个受检异常 ...

  2. java异常——捕获异常+再次抛出异常与异常链

    [0]README 0.1) 本文描述+源代码均 转自 core java volume 1, 旨在理解 java异常--捕获异常+再次抛出异常与异常链 的相关知识: [1]捕获异常相关 1.1)如果 ...

  3. 浅谈java异常[Exception]

    学习Java的同学注意了!!! 学习过程中遇到什么问题或者想获取学习资源的话,欢迎加入Java学习交流群,群号码:589809992 我们一起学Java! 一. 异常的定义 在<java编程思想 ...

  4. 基础知识《十》java 异常捕捉 ( try catch finally ) 你真的掌握了吗?

    本文转载自  java 异常捕捉 ( try catch finally ) 你真的掌握了吗? 前言:java 中的异常处理机制你真的理解了吗?掌握了吗?catch 体里遇到 return 是怎么处理 ...

  5. Java异常之自定义异常

    哎呀,妈呀,又出异常了!俗话说:"代码虐我千百遍,我待代码如初恋". 小Alan最近一直在忙着工作,已经很久没有写写东西来加深自己的理解了,今天来跟大家聊聊Java异常.Java异 ...

  6. 第11章 Java异常与异常处理

    1.Java异常简介 1.什么是异常异常出现的时候代码会无法正常运行下去,会产生各种问题2.捕捉异常的作用提早发现异常,方便查找问题,并给出解决方法3.Java中的异常1.Java中所有不正常的类都是 ...

  7. 3.Java异常进阶

    3.JAVA异常进阶 1.Run函数中抛出的异常 1.run函数不会抛出异常 2.run函数的异常会交给UncaughtExceptionhandler处理 3.默认的UncaughtExceptio ...

  8. 2.Java异常学习

    1.Java异常的概念 异常的例子 1.除法就是一个需要捕获异常的例子,除数又可能是0 异常处理的基本流程如下 一旦发生异常,就使得程序不按照原来的流程继续的运行下去 a.程序抛出异常 try{ th ...

  9. java异常架构图 和几个面试题

    1.java异常架构图 粉红色的是受检查的异常(checked exceptions),其必须被 try{}catch语句块所捕获,或者在方法签名里通过throws子句声明.受检查的异常必须在编译时被 ...

随机推荐

  1. vsftp虚拟用户方式访问

    需求:外部人员需要对公司服务器上某个文件夹内容进行读写操作 文件目录信息:/opt/abc drwxr-xr-x 9 www  www       4096 12月  4 13:02 abc   #注 ...

  2. Linux命令学习(4):gzip压缩与解压

    版权声明:本文为博主原创文章,未经允许不得转载 引子 gzip是Linux系统中最常用也是高效的压缩压缩命令.早期Linux系统中主要使用compress命令压缩,得到后缀为“.Z”的压缩文件,但是后 ...

  3. xtrbackup备份mysql

    mysqldump备份方式是采用逻辑备份,但是它最大的缺陷就是备份和恢复速度慢对于一个小于50G的数据库而言,这个速度还是能接受的,但如果数据库非常大,那再使用mysqldump备份就不太适合了. x ...

  4. asp.net:Parser Error & HTTP 错误 500.21 - Internal Server Error

    aaarticlea/png;base64,iVBORw0KGgoAAAANSUhEUgAABEcAAAF8CAIAAACiwUIQAAAgAElEQVR4nO2dyZHzOpZGaVEaADNkBB ...

  5. Python中的列表(6)

    列表切片 如何拿到列表中的部分元素,Python 引入了 “切片” 的概念. 上代码: words = ['a','b','c','d'] print(words[0:3]) console: 冒号( ...

  6. 集训第五周 动态规划 B题LIS

      Time Limit:1000MS     Memory Limit:32768KB     64bit IO Format:%I64d & %I64u Submit Status Des ...

  7. xcap发包工具的简单使用1(构造报文)

    xcap是一个免费的网络发包工具,可以构造和发送常用的网络报文,如arp,ip,icmp,udp等,支持构造报文和发送报文等. 报文隶属于报文组,每个报文组包含多个报文,因此,创建报文首先要创建报文组 ...

  8. BNUOJ 1541 Air Raid

    Air Raid Time Limit: 1000ms Memory Limit: 10000KB This problem will be judged on PKU. Original ID: 1 ...

  9. HDU 4941

    Magical Forest Problem Description There is a forest can be seen as N * M grid. In this forest, ther ...

  10. oracle将查询到的数据插入到数据库的表中

    一.Oracle数据库中,把一张表的查询结果直接生成并导入一张新表中.   例如:现有只有A表,查询A表,并且把结果导入B表中.使用如下SQL语句:   create table b as selec ...