前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >Java 的 Checked 和 Unchecked Exception

Java 的 Checked 和 Unchecked Exception

作者头像
大数据真好玩
发布2019-08-21 15:30:39
5.7K0
发布2019-08-21 15:30:39
举报
文章被收录于专栏:暴走大数据暴走大数据

本文作者:王蒙(Matt) http://matt33.com/2016/12/13/java-exception/

如果在 Java 应用中对 Exception 能够正确处理,那么将会使你的程序更具有健壮性。但是很多人对 Exception 中的 Checked ExceptionUnchecked Exception 并不理解,并且 Exception 又常常被被分为 JVM Exception 和程序 Exception,这就让一些开发者显得更加困惑了,本文就这几个概念详细讲述一下。

Checked Exception

Checked Exception 是必须在代码中进行恰当处理的 Exception,而且编译器会强制开发者对其进行处理,否则编译会不通过。你可以使用 catch 语句捕获这些 Exception 或者在方法声明处使用 throws 语句抛出该异常。

一般来说,Checked Exception 的发生主要是由于一些特殊情况没有考虑到,比如如果网络连接失败会抛出 IOException,但是我们的程序应该能够提前预料到这些可能发生的异常,并对其进行处理,这样程序在运行过程中才不会崩掉,这也是编译器强制开发者对 Checked Exception 进行处理的原因。假设在文件传输的过程中网络出现中断,这时候程序应该能够捕获到这种异常并进行处理(重新尝试传输文件)。

Unchecked Exception

Unchecked Exception 的发生有一些是由于开发者代码逻辑错误造成的,比如:NullPointerException 这种异常可以通过检查一个引用是否为 null 来进行避免。

但是也有一些 Unchecked Exception 出现并不是因为开发者程序的问题,这些 Exception 是 java.lang.Error 的子类。就像 OutOfMemoryError 可能发生在任意一个示例对象创建时,但我们不可能在每个对象实例创建时都使用 catch 块去捕获异常。因此,我们也就不可能预料这些异常的发生,编译器在编译时也无法检测到这些异常。

例子

下面这个例子,由于没有对 Checked Exception 进行处理而导致编译失败。

代码语言:javascript
复制
class UnhandledException {
    public static void main(String[] args) {
        throw new Exception();
    }
}

为了使上面的代码可以正确编译,我们可以在 try/catch 块中捕获相应的异常或者是使用 throws 在 main 方法声明处抛出异常。

但是如果在 main 方法内部抛出一个 Unchecked Exception,依然可以正常编译,下面的例子就可以正确编译。正如前面所述,Unchecked Exception 在编译期间是无法提前检测,因此,不对其进行处理也不会影响到正常编译。

代码语言:javascript
复制
class UnhandledException {
    public static void main(String[] args) {
        throw new NullPointerException();
    }
}

Exception 类层次结构

java.lang.Throwable 类是一个 Checked Exception,Java 的 API 定义了 Throwable 的两个子类——java.lang.Exceptionjava.lang.Error, Error 类是 Unchecked Exception 类,而 Exception 则是 Checked Exception类。

Exception 类有一个 Unchecked Exception 子类——java.lang.RuntimeException,NullPointerException 和 ClassCastException 都是 RuntimeException 的子类。RuntimeException 和 Error 的所有子类都是 Unchecked Exception,其他的 Exception 则都是 Checked Exception,如下图所示。

如果创建一个自定义的异常类,它是 Checked Exception 还是 Unchecked Exception 则依赖其父类的类型。如果它继承于一个 Unchecked Exception 类,那么它就是一个 Unchecked Exception,反之依然。

在对 Checked Exception 进行 catch 操作时,也需要遵循一定的规则:在 catch 块中捕获的异常,必须在 try 块中有出现这种异常的可能性。

代码语言:javascript
复制
try {
    System.out.println("...");
} catch(java.io.IOException e) {
}

上面的例子就不能成功编译,因为在 try 块中永远都不会抛出 IOException,所以你也不能去捕获这种异常。但是如果你捕获的是一个 Unchecked Exception,那么就不会有这种问题。

Exception 和 Throwable 这两个类有些特殊,虽然它们都是 Checked Exception 类,但你依然可以捕获它们即使在 try 块中没有抛出该异常的可能性,因此,下面的代码的就可以正确编译。

代码语言:javascript
复制
try {
    System.out.println("...");
} catch(Exception ex) {
}

前面所说的规则对于 Exception 和 Throwable 这两个类并不是完全适用,这是因为对 Exception 和 Throwable 这两个类都有 Unchecked Exception 类型的子类,所以编译器允许你捕获它们(编译器认为你是在捕获一个 Unchecked Exception)。要清楚一点,编译器并不会检查 Unchecked Exception,RuntimeException 是 Exception 的子类,Error 是 Throwable的子类, 而 RuntimeException 和 Error 都是 Unchecked Exception 类。因此,上面的代码是可以正确编译的,编译允许这样做的原因就是因为这种方式是可以捕获到 Unchecked Exception 的。

JVM 和程序异常

JVM Exception 是由 JVM 自己抛出的异常,比如:如果调用的方法使用一个 null 引用,然后 JVM 就会抛出 NullPointerException,或者如果在程序中出现10除以0的情况,JVM 会抛出一个 ArithmeticException。这些异常都是自动地由 JVM 抛出。

除了 JVM Exception 外,其他所有的异常都是由程序引起的异常。程序中,我们可以显式地使用 throw语句抛出异常,这里以 NumberFormatException 为例。NumberFormatException 可能被方法 Integer.parseIntFloat.parseFloat 抛出,都是程序中可能出现的异常。在 Integer 类方法 parseInt 的实现中,可以找到如下的声明:

代码语言:javascript
复制
if (s == null) {
      throw new NumberFormatException("null");
}

而 JVM 不会抛出这种类型的异常,这些异常是使用 throw 语句显式地程序中抛出。当然也可以如下所示在程序中抛出 JVM Exception。

代码语言:javascript
复制
if (s == null) {
      throw new NullPointerException("I told you s shouldn't be null");
}

但是一般情况下,JVM Exception 是不会被开发者抛出的(JVM 自己抛出的),所有的 JVM Exception 都是 unchecked,而程序中的异常则可能是 checked 的或者 unchecked 的。

本文参与 腾讯云自媒体分享计划,分享自微信公众号。
原始发表:2019-08-18,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 大数据真好玩 微信公众号,前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体分享计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • Checked Exception
  • Unchecked Exception
  • 例子
  • Exception 类层次结构
  • JVM 和程序异常
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档