资讯专栏INFORMATION COLUMN

Java™ 教程(如何抛出异常)

zhangwang / 2905人阅读

如何抛出异常

在捕获异常之前,某些代码必须抛出一个,任何代码都可能抛出异常:你的代码,来自其他人编写的包中的代码,例如Java平台附带的包或Java运行时环境,无论抛出什么异常,它总是使用throw语句抛出。

你可能已经注意到,Java平台提供了许多异常类,所有类都是Throwable类的后代,并且所有类都允许程序区分在程序执行期间可能发生的各种类型的异常。

你还可以创建自己的异常类来表示你编写的类中可能出现的问题,实际上,如果你是程序包开发人员,则可能必须创建自己的一组异常类,以允许用户将程序包中可能发生的错误与Java平台或其他程序包中发生的错误区分开来。

你还可以创建链式异常,有关更多信息,请参阅“链式异常”部分。

throw语句

所有方法都使用throw语句抛出异常,throw语句需要一个参数:一个throwable对象,Throwable对象是Throwable类的任何子类的实例,这是一个throw语句的例子。

throw someThrowableObject;

让我们看一下上下文中的throw语句,以下pop方法取自实现公共堆栈对象的类,该方法从堆栈中删除顶部元素并返回该对象。

public Object pop() {
    Object obj;

    if (size == 0) {
        throw new EmptyStackException();
    }

    obj = objectAt(size - 1);
    setObjectAt(size - 1, null);
    size--;
    return obj;
}

pop方法检查堆栈上是否有任何元素,如果堆栈为空(其大小等于0),则pop实例化一个新的EmptyStackException对象(java.util的成员)并抛出它,本章中的创建异常类部分介绍了如何创建自己的异常类,现在,你需要记住的是,你只能抛出从java.lang.Throwable类继承的对象。

请注意,pop方法的声明不包含throws子句,EmptyStackException不是已检查的异常,因此不需要pop来声明它可能发生。

Throwable类及其子类

Throwable类继承的对象包括直接后代(直接从Throwable类继承的对象)和间接后代(从Throwable类的子级或孙级继承的对象),下图说明了Throwable类的类层次结构及其最重要的子类,如你所见,Throwable有两个直接后代:Error和Exception。

Error类

当发生Java虚拟机中的动态链接故障或其他硬故障时,虚拟机抛出Error,简单程序通常不会捕获或抛出Errors

Exception类

大多数程序抛出并捕获从Exception类派生的对象,Exception表示发生了问题,但这不是一个严重的系统问题,你编写的大多数程序将抛出并捕获Exception而不是Error

Java平台定义了Exception类的许多后代,这些后代表示可能发生的各种类型的异常。例如,IllegalAccessException表示无法找到特定方法,NegativeArraySizeException表示程序试图创建负大小的数组。

一个Exception子类RuntimeException保留用于指示错误使用API​​的异常,运行时异常的一个示例是NullPointerException,当方法尝试通过空引用访问对象的成员时发生,未经检查的异常 — 争议部分讨论了为什么大多数应用程序不应抛出运行时异常或RuntimeException子类。

链式异常

应用程序通常会通过抛出另一个异常来响应异常,实际上,第一个异常导致第二个异常,了解一个异常何时导致另一个异常非常有用,链式异常有助于程序员执行此操作。

以下是Throwable中支持链式异常的方法和构造函数。

Throwable getCause()
Throwable initCause(Throwable)
Throwable(String, Throwable)
Throwable(Throwable)

initCauseThrowable构造函数的Throwable参数是导致当前异常的异常,getCause返回导致当前异常的异常,initCause设置当前异常的原因。

以下示例显示如何使用链式异常。

try {

} catch (IOException e) {
    throw new SampleException("Other IOException", e);
}

在此示例中,捕获IOException时,会创建一个新的SampleException异常,并附加原始原因,并将异常链抛出到下一个更高级别的异常处理程序。

访问堆栈跟踪信息

现在让我们假设更高级别的异常处理程序想要以自己的格式转储堆栈跟踪。

定义:堆栈跟踪提供有关当前线程的执行历史记录的信息,并列出在发生异常时调用的类和方法的名称,堆栈跟踪是一种有用的调试工具,通常在抛出异常时可以利用它。

以下代码显示如何在异常对象上调用getStackTrace方法。

catch (Exception cause) {
    StackTraceElement elements[] = cause.getStackTrace();
    for (int i = 0, n = elements.length; i < n; i++) {       
        System.err.println(elements[i].getFileName()
            + ":" + elements[i].getLineNumber() 
            + ">> "
            + elements[i].getMethodName() + "()");
    }
}
Logging API

下一个代码段记录catch块中发生异常的位置,但是,它不是手动解析堆栈跟踪并将输出发送到System.err(),而是使用java.util.logging包中的日志记录工具将输出发送到文件。

try {
    Handler handler = new FileHandler("OutFile.log");
    Logger.getLogger("").addHandler(handler);
    
} catch (IOException e) {
    Logger logger = Logger.getLogger("package.name"); 
    StackTraceElement elements[] = e.getStackTrace();
    for (int i = 0, n = elements.length; i < n; i++) {
        logger.log(Level.WARNING, elements[i].getMethodName());
    }
}
创建异常类

当面对选择要抛出的异常类型时,你可以使用其他人编写的异常 — Java平台提供了许多可以使用的异常类 — 或者你可以编写自己的异常类,如果你对以下任何问题的回答是肯定的,你应该编写自己的异常类;否则,你可能会使用别人的。

你是否需要Java平台中未表示的异常类型?

如果他们可以将你的异常与其他供应商编写的类别所引发的异常区分开来,它会帮助用户吗?

你的代码是否会抛出多个相关的异常?

如果你使用其他人的异常,用户是否可以访问这些异常?一个类似的问题是,你的包是否应该独立且自包含?

一个例子

假设你正在编写链表类,该类支持以下方法,其中包括:

objectAt(int n) — 返回列表中第n个位置的对象,如果参数小于0或大于列表中当前对象的数量,则引发异常。

firstObject() — 返回列表中的第一个对象,如果列表不包含任何对象,则抛出异常。

indexOf(Object o) — 在列表中搜索指定的Object并返回其在列表中的位置,如果传递给方法的对象不在列表中,则抛出异常。

链表类可以抛出多个异常,并且能够通过一个异常处理程序捕获链表所引发的所有异常会很方便,此外,如果你计划在包中分发链表,则应将所有相关代码打包在一起,因此,链表应该提供自己的一组异常类。

下图说明了链表抛出的异常的一个可能的类层次结构。

选择超类

任何Exception子类都可以用作LinkedListException的父类,但是,快速浏览这些子类就会发现它们不合适,因为它们太专业化或与LinkedListException完全无关,因此,LinkedListException的父类应该是Exception

你编写的大多数applet和应用程序都会抛出Exception对象,Error通常用于系统中严重的硬错误,例如阻止JVM运行的错误。

对于可读代码,最好将字符串Exception附加到从Exception类继承(直接或间接)的所有类的名称。
上一篇:捕获和处理异常

文章版权归作者所有,未经允许请勿转载,若此文章存在违规行为,您可以联系管理员删除。

转载请注明本文地址:https://www.ucloud.cn/yun/77477.html

相关文章

  • Java教程(捕获和处理异常

    捕获和处理异常 本节描述如何使用三个异常处理程序组件 — try、catch和finally块 — 来编写异常处理程序,然后,解释了Java SE 7中引入的try-with-resources语句,try-with-resources语句特别适用于使用Closeable资源的情况,例如流。 本节的最后一部分将介绍一个示例,并分析各种场景中发生的情况。 以下示例定义并实现名为ListOfNumbe...

    Yujiaao 评论0 收藏0
  • Java教程异常的优点)

    异常的优点 现在你已经知道了什么是异常以及如何使用它们,现在是时候了解在程序中使用异常的优势了。 优点1:将错误处理代码与常规代码分开 异常提供了从程序的主逻辑中分离异常发生时应该做什么的细节的方法,在传统的编程中,错误检测、报告和处理通常会导致混乱的意大利面代码,例如,考虑这里的伪代码方法将整个文件读入内存。 readFile { open the file; determine...

    jollywing 评论0 收藏0
  • Java教程(捕获或指定要求)

    捕获或指定要求 有效的Java编程语言代码必须遵守捕获或指定需求,这意味着可能抛出某些异常的代码必须包含以下任一项: 捕获异常的try语句,try必须为异常提供处理程序,如捕获和处理异常中所述。 一种方法,指定它可以抛出异常,该方法必须提供一个throws子句,列出异常,如通过方法抛出指定异常中所述。 不符合捕获或指定要求的代码将无法编译。 并非所有异常都受捕获或指定要求的约束,为了理解原因,...

    wanglu1209 评论0 收藏0
  • Spring Boot 2.x 系列教程:WebFlux REST API 全局异常处理 Error

    摘要:挺多人咨询的,异常处理用切面注解去实现去全局异常处理。全局异常处理类,代码如下代码解析如下抽象类是用来处理全局错误时进行扩展和实现注解标记的切面排序,值越小拥有越高的优先级,这里设置优先级偏高。 本文内容 为什么要全局异常处理? WebFlux REST 全局异常处理实战 小结 摘录:只有不断培养好习惯,同时不断打破坏习惯,我们的行为举止才能够自始至终都是正确的。 一、为什么要全局...

    BicycleWarrior 评论0 收藏0
  • 猫头鹰的深夜翻译:JAVA异常处理的最佳实践

    摘要:无需检查的异常也是的子类。从低层抛出的需检查异常强制要求调用方捕获或是抛出该异常。当前执行的线程将会停止并报告该异常。单元测试允许我在使用中查看异常,并且作为一个可以被执行的文档来使用。不要捕获最高层异常继承的异常同样是的子类。 前言 异常处理的问题之一是知道何时以及如何去使用它。我会讨论一些异常处理的最佳实践,也会总结最近在异常处理上的一些争论。 作为程序员,我们想要写高质量的能够解...

    W_BinaryTree 评论0 收藏0

发表评论

0条评论

zhangwang

|高级讲师

TA的文章

阅读更多
最新活动
阅读需要支付1元查看
<