Java中的异常

本文基于JDK 1.8.0_45

Throwable架构是非常经典的继承结构:

  1. 所有的广意的异常都是Throwable的子类,通过继承的树状结构来对异常进行分类,比如异常分为两大类,Error和Exception,其中Exception又分为检查异常和运行期异常,每一个分支又是细化的一类异常。
  2. 在Throwable中定义大多数关于异常处理的方法,并提供部分接口给子类重写。
  3. 一般都可以从异常命名中看出来其作用是什么,因此当我们自定义异常的时候也最好遵循这一命名规则。
  4. printStackTrace方法有PrintWriter和PrintStream两种参数的重载方法,源码中通过自定义的PrintStreamOrWriter内部类的两种子类分别对这两种类型的参数进行封装,并通过统一接口提供lock和println的方法,通过这种方式避免了复制代码来分别实现相似的功能。该方法通过递归遍历并打印自己的异常堆栈、suppressed异常堆栈和cause异常堆栈。
  5. 从1.7版本开始添加了两个API方法:addSuppressed和getSuppressed来解决存在多个异常抛出的场景(如try-with-resources)。其中try-with-resources的字节码大概是下面的样子,有可能会出现两个异常,一个是try块中的异常,一个是调用close方法时抛出的异常。使用suppressed就可以避免前一个异常被丢弃而无法找到真正的异常原因的问题。
try{
  throw new Throwable();
}catch(Throwable e) {
  try {
    resource.close();
  }catch(Throwable suppressedException) {
    e.addSuppressed(suppressedException);
    throw e;
  }
  throw suppressedException;
}

Throwable

Throwable提供五种构造方法来构造Throwable,可以传入具体信息,发生原因等,一般异常均为根据需要继承实现这五种构造方法或某几种构造方法,如Exception和Error均为直接实现这五种构造方法。

public Throwable() {...}

public Throwable(String message) {...}

public Throwable(String message, Throwable cause) {...}

public Throwable(Throwable cause) {...}

public Throwable(String message, Throwable cause, boolean enableSuppression, boolean writableStackTrace) {...}

另外Throwable还提供了丰富的方法,比如getLocalizedMessage让异常提供本地化的异常信息,很多StackTrace相关的方法让用户可以很好的获取、操作StackTrace。

WARNING, 有时候我们会通过getCause或getSuppressed方法来做一些个性化的处理,由于该方法会返回一个新的Throwable对象,因此在处理过程中经常会使用递归或循环的方式遍历到尽头,但是在遍历的过程中一定要小心循环依赖的问题,否则会导致系统资源耗尽而出严重问题。从以下代码在执行到exception1.printStackTrace();时的打印信息中可以看到CIRCULAR REFERENCE,这是因为exception2在new的时候指定它的cause是exception1,而在exception1.initCause(exception2);又指定exception1的cause是exception2,从而出现了异常cause的循环依赖。

package com.oomlife.java.example;

public class ExceptionExample {
    public static void main(String[] args) {
        FakeException exception1 = new FakeException();
        FakeFakeException exception2 = new FakeFakeException(exception1);
        exception1.initCause(exception2);

        exception1.printStackTrace();
    }

    public static class FakeException extends RuntimeException {
    }

    public static class FakeFakeException extends RuntimeException {
        public FakeFakeException(Throwable cause) {
            super(cause);
        }
    }
}


打印结果:
com.oomlife.java.example.ExceptionExample$FakeException
    at com.oomlife.java.example.ExceptionExample.main(ExceptionExample.java:5)
    at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
    at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62)
    at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
    at java.lang.reflect.Method.invoke(Method.java:497)
    at com.intellij.rt.execution.application.AppMain.main(AppMain.java:147)
Caused by: com.oomlife.java.example.ExceptionExample$FakeFakeException: com.oomlife.java.example.ExceptionExample$FakeException
    at com.oomlife.java.example.ExceptionExample.main(ExceptionExample.java:6)
    ... 5 more
    [CIRCULAR REFERENCE:com.oomlife.java.example.ExceptionExample$FakeException]

StackTraceElement类是异常堆栈信息的类,它提供了所在class、所在方法、所在文件名和行号等信息,一般每个实例是异常堆栈中的一行,当使用Java实现自己的DSL的时候,这个类对实现自己的异常跟踪和打印有很大的借鉴意义。

Exception

Java中的异常分为两种,受检查的异常和运行期异常。

受检查的异常在代码中必须进行显式声明或处理,否则代码无法编译通过,比如SQLException。

运行期的异常是指在Java程序执行的过程中由于一些特殊的情况导致的异常,比如ClassCastException是将一个对象强制转为另一个不相关的对象时抛出的异常,再比如ArrayIndexOutOfBoundsException是在运行期访问数组的一个非法索引时抛出的异常,这些异常大都是在写代码的时候很难预判或进行预先检查的成本太高,比如NullPointerException。

Java API中提供了种类繁多的异常类型,在实际应用中,我们可以直接使用Java API中提供的异常,也可以自定义异常。在自定义异常的时候最好遵从Java API中的一些最佳实践:

  1. 将自定义异常区分为受检查的异常和运行期异常;
  2. 如果需要将受检查异常捕获并重新抛出运行期异常时要慎重考虑是否真的有必要将异常处理延后到运行期;
  3. 在重写的构造方法中一定要恰当调用父类的构造方法来封装异常的详细信息,避免添加重复的字段;
  4. 异常命名要是自描述的,从名字即可很容易的看出来异常的用途是什么。

Error


Error具有和Exception相同的结构,一般用于指代那些严重的无法被处理的错误,如运行时编译错误,系统错误等,以下是两个Java内置错误的例子:

LinkageError


这个错误是class相关的错误,比如在编译成功后class被错误修改了,有环形引用等。这是一个受检查的错误,主要是被编译器捕获。另外如果一个class的定义在当前执行方法最后一次编译以后作了不兼容的更改,则此错误可能在运行期发生。包括ExceptionInInitializerError、NoClassDefFoundError、IncompatibleClassChangeError、ClassCircularityError、ClassFormatError(包括GenericSignatureFormatError、UnsupportedClassVersionError)、VerifyError、BootstrapMethodError和UnsatisfiedLinkError,其中IncompatibleClassChangeError系列包括AbstractMethodError、IllegalAccessError、InstantiationError、NoSuchFieldError和NoSuchMethodError。下面是一个下运行期的AbstractMethodError的例子:

public class AbstractClass {
  public void hello() {
    System.out.println("Hello, I'm an abstract class.");
  }
}
public class MainClass extends AbsClass {
  public static void main(String[] args) {
    MainClass cl = new MainClass();
    cl.hello();
  }
}

比如我先编译以上两个class并执行,结果如下:

> javac AbstractClass.java
> javac MainClass.java
> java MainClass
Hello, I'm an abstract class.

一切正常,然后我们修改AbstractClass如下:

public class AbstractClass {
  public abstract void hello();
}

重新编译AbstractClass并执行,结果如下:

> javac AbstractClass.java
> java MainClass
Exception in thread "main" java.lang.AbstractMethodError:
MainClass.hello()V
        at MainClass.main(MainClass.java:4)

VirtualMachineError


这是JVM相关的错误,包括OutOfMemoryError(发生于内存不足的情况)、StackOverflowError(比如非尾递归的递归方法调用的时候可能会发生)、InternalError,还有其他未知错误UnknownError。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,335评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,895评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,766评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,918评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,042评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,169评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,219评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,976评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,393评论 1 304
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,711评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,876评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,562评论 4 336
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,193评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,903评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,142评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,699评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,764评论 2 351

推荐阅读更多精彩内容

  • 如果我们制作一个计算器程序,当用户输入的除数为0时,程序将会崩溃直接退出,那么该程序的用户体验将会非常差。我们应该...
    yuluo阅读 582评论 0 1
  • 两类异常 Java中的异常继承体系为: 这里的异常( Exception)分为两大类:Checked异常和Runt...
    tobe_superman阅读 634评论 0 0
  • 我们通过一个简单的实例程序来了解一下什么是java中的异常处理 使用try,catch 看下面这个程序: 这个程序...
    六尺帐篷阅读 1,342评论 0 7
  • 数十年来,江湖上从没有人敢悖逆理发六十的规矩,登哪山,拜哪门,你吃这碗饭,就得先给自己立这个槛。 可一年前彼得在人...
    将要远行阅读 660评论 1 2
  • 昨天夜里,大约两点钟的时候。隔壁床的姑娘把我叫醒了,说院子里有动静。我仔细一听,确实有什么东西在呼呼作响。...
    二乔木阅读 116评论 0 0