Java异常处理如何使用


这篇文章主要介绍了Java异常处理如何使用的相关知识,内容详细易懂,操作简单快捷,具有一定借鉴价值,相信大家阅读完这篇Java异常处理如何使用文章都会有所收获,下面我们一起来看看吧。异常处理的概念起源于早期的编程语言,如 LISP、PL/I 和 CLU。这些编程语言首次引入了异常处理机制,以便在程序执行过程中检测和处理错误情况。异常处理机制随后在 Ada、Modula-3、C++、Python、Java 等编程语言中得到了广泛采用和发展。在 Java 中,异常处理是提供一种在程序运行时处理错误和异常情况的方法。异常处理机制使得程序能够在遇到错误时继续执行,而不是立即崩溃。这种机制使程序更具有健壮性和容错性。异常分为两类:受检异常(Checked Exceptions)和非受检异常(Unchecked Exceptions)受检异常(Checked Exceptions):受检异常是指那些在编译时必须处理的异常。它们通常是由程序员的错误或外部资源问题引起的。例如,IOExceptionFileNotFoundException等。受检异常必须在方法签名中使用throws关键字声明,或者在方法体内用try-catch块捕获和处理。非受检异常(Unchecked Exceptions):非受检异常是指那些在编译时不强制要求处理的异常。它们通常是由编程错误引起的,如空指针异常(NullPointerException)、数组越界(ArrayIndexOutOfBoundsException)等。非受检异常继承自java.lang.RuntimeException类,不需要在方法签名中声明,也不需要强制捕获和处理。它们的关系如下:Java 使用try/catch关键字进行异常捕获,使用throw声明抛出异常,示例代码如下:在这个示例中,我们尝试获取一个null字符串的长度。当调用nullString.length()时,会抛出NullPointerException。我们使用try-catch语句捕获异常并处理它Java 官方异常不可能预见所有可能发生的错误,有时候你需要结合自己的业务场景,例如以下场景:当内置的 Java 异常类无法准确描述你所遇到的异常情况时。需要为特定领域或业务逻辑创建一组特定的异常。当希望通过自定义异常类向调用者提供更多的上下文信息或特定的错误代码时。构建特定的异常,这也很简单,继承已有的异常类(最好继承含义差不多的),如下,我们创建一个表示账户余额不足的异常:接下来,我们在业务逻辑代码中使用这个自定义异常:调用者可以捕获并处理这个自定义异常:可以看到自定义异常使我们能够更清晰地表达业务逻辑中可能出现的异常情况,同时为调用者提供更多关于异常的上下文信息。我们还使用java.util.logging工具将输出记录到日志中在 Java 早起版本,处理多个没有共同基类的异常,需要为每一个异常类型编写一个 catch 语句处理,如下:这样的代码不仅难以阅读,而且也不够简洁。多重异常捕获机制,它允许在一个catch语句中捕获多个异常类型。这种方法可以避免重复的代码,使异常处理更加简洁。以下是一个使用多重异常捕获机制的示例:在某些情况下,你可能希望将异常传递给调用者处理,而不是在当前方法中处理。或者需要在捕获异常时执行一些处理操作,如记录日志、清理资源或者添加额外的上下文信息。在这种情况下,你可以在catch块中处理异常,然后重新抛出原始异常或抛出一个新的异常,包含额外的信息,如下:NPE(NullPointerExceptions)是很常见异常,在 JDK 14 以前遇到 NPE 异常,能得到信息有限,JDK 15 引入了一项名为“Helpful NullPointerExceptions”的新功能,这项功能改进了 NullPointerException (NPE) 的诊断。在之前的 JDK 版本中,当发生 NullPointerException 时,异常信息通常并不提供足够的上下文来帮助开发人员定位问题的具体位置。示例代码:在 JDK 14 及之前的版本中,输出结果:# 你根本看不出哪里出了问题
null
java.lang.NullPointerException
java.lang.NullPointerException在 JDK 15 及之后的版本中,输出结果:# 得到更详细的 NPE 信息
null
java.lang.NullPointerException: Cannot read field “s” because “c.b.a” is null
java.lang.NullPointerException: Cannot read field “a” because “c.b” is null当程序发生了非预期的异常,那么程序会终止运行,但是对于很多需要执行清理操作,这是不可接受的,例如:关闭资源:在try块中打开的资源,如文件、数据库连接、网络连接等,需要在完成操作后确保被正确关闭释放锁:在并发编程中,可能会使用锁来同步代码。在释放锁之前,如果发生异常,可能会导致其他线程无法获取锁回滚事务:在数据库编程中,可能需要在事务中执行一系列操作。如果这些操作中的任何一个失败,事务需要回滚恢复状态:在执行某些操作时,可能需要更改对象或系统的状态。在操作完成后,可能需要恢复原始状态对于以上的程序来说,finally 就非常重要了,它可以解决以上程序的清理操作。示例代码:在以上代码中,无论程序是否出错,finally都可以确保文件被正确关闭Java 在面向对象中对异常存在颇多约束和限制,其主要目的如下:保持子类型可替换性:当子类覆盖父类的方法或实现接口的方法时,子类的方法应该满足父类或接口方法的约定避免意外的异常:如果子类方法可以抛出任意异常,那么调用者在处理异常时可能遇到意外的异常类型,导致程序出错提高代码可读性:通过限制异常的继承和实现规则,可以使得代码更加清晰和易于理解促进良好的设计实践:如果子类方法可以抛出任意异常,那么程序员可能会过度依赖异常来处理错误情况,导致代码难以维护在接口和继承中使用异常,需要遵循以下规则:子类可以抛出与接口或者父类方法相同的异常。子类可以不抛出任何异常,即使接口或父类方法声明了异常。这意味着实现类的方法已经处理了这些异常。子类可以抛出接口方法或父类声明异常的相同类型的异常,因为子类异常依然符合接口方法的约定。代码示例:在 Java 7 中,关于自动管理资源。在处理需要关闭的资源(如文件、数据库连接、网络连接等)有了更好的选择,那就是使用try-catch-finally进行处理,它对比finally具有以下优势:简化代码:相比finally显示关闭资源,使用 Try-With-Resources,可以自动关闭资源,从而使代码更简洁、易读。避免资源泄漏:使用 Try-With-Resources 能够确保在退出try代码块时自动关闭资源,降低资源泄漏的风险。减少错误:Try-With-Resources 能够正确地处理资源关闭过程中的异常,并提供完整的异常信息,有助于减少错误。示例代码:在这个示例中,我们使用 Try-With-Resources 语句创建了一个BufferedReader实例。BufferedReader实现了Closeable接口,因此在退出try代码块时,reader会自动调用close()方法以释放资源。为了一探 Try-With-Resources 的究竟,我们可以创建自定义的AutoCloseable类:输出结果:Creating First
Creating Second
In body
Closing Second
Closing First退出 try 块会调用两个对象的 close() 方法,并以与创建顺序相反的顺序关闭它们。(顺序很重要)。使用 Try-With-Resource 是很安全的,假设你随意在 Try 头使用对象,会出现编译错误:Java 在抛出异常时,会根据异常类型进行匹配。异常处理程序会从上到下依次检查catch子句,看它们是否与抛出的异常类型兼容。当发现兼容的catch子句时,Java 就会执行该子句的代码来处理异常。请注意,Java 只会执行与抛出异常类型兼容的第一个catch子句。示例代码:在这个示例中,我们抛出了一个FileNotFoundException,Java 会从上到下检查catch子句,看它们是否与FileNotFoundException兼容。因为FileNotFoundExceptionIOException的子类,它与FileNotFoundExceptionIOExceptioncatch子句兼容。但是,Java 只会执行第一个兼容的catch子句,即FileNotFoundException子句。如果没有找到兼容的catch子句,Java 会继续在调用栈中查找异常处理程序,直到找到一个合适的处理程序或者程序终止。输出结果:Handling FileNotFoundException: File not found异常看似简单易懂,但在处理过程中还需要遵循许多最佳实践,例如:除非你知道如何处理,否则不要捕获异常(错误处理代码太多,容易干扰主线代码的逻辑和可读性)不要生吞异常:捕获异常不进行处理会导致异常消失,从而对于线上问题排查,无从下免费云主机域名手捕获具体异常:尽量捕获具体的异常类,而不是捕获泛化的 Exception 类尽可能的使用多重异常捕获来简化重复代码,并且提高代码的可读性尽可能的使用 Try-With-Resources 清理资源自定义异常:在需要时,为特定于你的应用程序的异常情况创建自定义异常检查型异常(checked exceptions)在 Java 中引发了很多争议。有些人认为它们是一种有益的设计,可以提高代码的可靠性,而另一些人则认为它们是一种糟糕的设计,会导致代码冗余和难以维护,例如 Martin Fowler (《UML 精粹》、《重构》)作者,也曾在博客发表称:总的来说,我认为异常很不错,但是 Java 的检查型异常要比好处多那么检查型异常究竟带来了什么问题 ? 常见的槽点有:强制错误处理:检查型异常强制开发者处理异常情况,导致主线代码中充斥着大量和业务逻辑无关的代码代码冗余:检查型异常可能导致大量的 try-catch 代码块,增加代码冗余,影响代码可读性异常传递:对于一些需要在多层方法调用中传递异常的情况,检查型异常可能导致开发者不得不为每个方法添加异常声明最几年 Go 语言的成功让很多人加深了这一观点,Go 语言没有检查型异常的概念,但它们的代码依然可以具有很高的可靠性。这表明检查型异常并非是提高代码可靠性的唯一方法。Go 语言的设计者们有意避免了引入检查型异常,主要有以下原因:代码简洁性:Go 语言的设计者们希望保持代码简洁,避免因为异常处理而产生的大量冗余代码。显示错误处理:Go 语言鼓励开发者显式地处理错误,而不是通过异常机制隐式地处理。降低复杂性:异常机制会增加程序的复杂性。Go 语言的设计者们希望通过避免引入异常机制,让程序更简单易懂。性能开销:异常处理机制可能会带来一定的性能开销,通过返回 error 类型,可以避免这种性能开销。示例代码:在这个示例中,我们定义了一个名为divide的函数,它接受两个整数参数ab,计算它们的商。如果b为零,函数将返回一个非空的error类型值,以指示发生了错误。否则,函数将返回商和一个空error值。在main函数中,我们调用divide两次,一次使用一个非零除数,另一次使用零作为除数。对于第一次调用,divide将返回一个空的error值,我们就可以打印出计算结果。对于第二次调用,divide将返回一个非空的error值,我们使用if err != nil来检查这个值是否为nil,如果不是,就打印错误信息。输出结果:Result: 5
Error: division by zero关于“Java异常处理如何使用”这篇文章的内容就介绍到这里,感谢各位的阅读!相信大家对“Java异常处理如何使用”知识都有一定的了解,大家如果还想学习更多知识,欢迎关注百云主机行业资讯频道。

相关推荐: 怎么用java判断素数

今天小编给大家分享一下怎么用java判断素数的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文章给大家参考一下,希望大家阅读完这篇文章后有所收获,下面我们一起来了解一下吧。 1、判断素数的方法:用一个数分别去除2到sqrt(这…

免责声明:本站发布的图片视频文字,以转载和分享为主,文章观点不代表本站立场,本站不承担相关法律责任;如果涉及侵权请联系邮箱:360163164@qq.com举报,并提供相关证据,经查实将立刻删除涉嫌侵权内容。

Like (0)
Donate 微信扫一扫 微信扫一扫
Previous 07/13 09:39
Next 07/13 09:39

相关推荐