Java中的异常处理是一个非常有趣的主题。异常是在程序执行期间可能发生的错误事件,并且会中断其正常流程。Java提供了一种强大的面向对象的方法来处理异常场景,称为Java异常处理。我们将在本教程中研究以下主题。
我们不喜欢异常,但我们总是要处理它们,好消息是Java中的异常处理非常强大且易于理解和使用。java中的异常可能来自不同类型的情况,例如用户输入的错误数据,硬件故障,网络连接故障,数据库服务器故障等。在本节中,我们将学习如何在java中处理异常。
Java是一种面向对象的编程语言,只要在执行语句时发生错误,就会创建一个异常对象,然后程序的正常流程停止,JRE会尝试找到可以处理引发的异常的人。异常对象包含许多调试信息,例如方法层次结构,发生异常的行号,异常类型等。当方法中发生异常时,将创建异常对象并将其交给运行时环境的过程调用“抛出异常”。
运行时收到异常对象后,会尝试查找异常的处理程序。异常处理程序是可以处理异常对象的代码块。查找异常处理程序的逻辑很简单 - 在发生错误的方法中开始搜索,如果找不到合适的处理程序,则转移到调用方法等等。因此,如果方法调用堆栈是A-> B-> C并且在方法C中引发异常,则搜索适当的处理程序将从C-> B-> A移动。如果找到适当的异常处理程序,则将异常对象传递给处理程序以对其进行处理。据说处理程序是“捕获异常”。如果找不到合适的异常处理程序,则程序终止打印有关异常的信息。
请注意,Java Exception处理是一个仅用于处理运行时错误的框架,java中的异常处理不处理编译时错误。
我们在java程序中使用特定关键字来创建异常处理程序块,接下来我们将研究这些关键字。
Java为异常处理目的提供了特定的关键字,我们将首先关注它们,然后我们将编写一个简单的程序,展示如何使用它们进行异常处理。
让我们看一个简单的编程,显示java中的异常处理。
package com.journaldev.exceptions;
import java.io.FileNotFoundException;
import java.io.IOException;
public class ExceptionHandling {
public static void main(String[] args) throws FileNotFoundException, IOException {
try{
testException(-5);
testException(-10);
}catch(FileNotFoundException e){
e.printStackTrace();
}catch(IOException e){
e.printStackTrace();
}finally{
System.out.println("Releasing resources");
}
testException(15);
}
public static void testException(int i) throws FileNotFoundException, IOException{
if(i < 0){
FileNotFoundException myException = new FileNotFoundException("Negative Integer "+i);
throw myException;
}else if(i > 10){
throw new IOException("Only supported for index 0 to 10");
}
}
}
上述计划的输出是:
java.io.FileNotFoundException: Negative Integer -5
at com.journaldev.exceptions.ExceptionHandling.testException(ExceptionHandling.java:24)
at com.journaldev.exceptions.ExceptionHandling.main(ExceptionHandling.java:10)
Releasing resources
Exception in thread "main" java.io.IOException: Only supported for index 0 to 10
at com.journaldev.exceptions.ExceptionHandling.testException(ExceptionHandling.java:27)
at com.journaldev.exceptions.ExceptionHandling.main(ExceptionHandling.java:19)
请注意,testException()方法使用throw关键字抛出异常,方法签名使用throws关键字让调用者知道它可能抛出的异常类型。在main()方法中,我在main()方法中使用try-catch块处理异常,当我不处理它时,我在main方法中使用throws子句将其传播到运行时。请注意,testException(-10)
永远不会因异常而执行,然后在执行try-catch块后执行finally块。printStackTrace()是Exception类中有用的方法之一,用于调试目的。
如前所述,当引发任何异常时,将创建异常对象。Java异常是分层的,继承用于对不同类型的异常进行分类。Throwable是Java Exceptions Hierarchy的父类,它有两个子对象 - Error和Exception。异常进一步分为检查异常和运行时异常。
ArrayIndexOutOfBoundException
在运行时抛出。RuntimeException是所有运行时异常的父类。如果我们在方法中抛出任何运行时异常,则不需要在方法签名throws子句中指定它们。通过更好的编程可以避免运行时异常。
Java异常及其所有子类不提供任何特定方法,并且所有方法都在基类Throwable中定义。创建异常类以指定不同类型的异常场景,以便我们可以轻松识别根本原因并根据其类型处理异常。Throwable类实现Serializable接口以实现互操作性。
Throwable类的一些有用方法是;
getMessage()
方法返回异常消息。
如果你在一个try块中捕获了很多异常,你会发现catch块代码看起来非常难看,而且大多数都是由冗余代码来记录错误,记住Java 7的一个特性是改进了catch块的地方我们可以在一个catch块中捕获多个异常。具有此功能的catch块如下所示:
catch(IOException | SQLException ex){
logger.error(ex);
throw new MyException(ex.getMessage());
}
有一些约束,例如异常对象是final,我们无法在catch块内修改它,在Java 7 Catch Block Improvements中读取完整分析。
大多数情况下,我们使用finally块来关闭资源,有时我们忘记关闭它们并在资源耗尽时获得运行时异常。这些异常很难调试,我们可能需要查看我们使用该类资源的每个地方,以确保我们关闭它。所以java 7的改进之一就是try-with-resources,我们可以在try语句中创建一个资源,并在try-catch块中使用它。当执行来自try-catch块时,运行时环境会自动关闭这些资源。具有这种改进的try-catch块样本是:
try (MyResource mr = new MyResource()) {
System.out.println("MyResource created in try-with-resources");
} catch (Exception e) {
e.printStackTrace();
}
在Java 7自动资源管理中阅读此功能的详细说明。
Java提供了许多异常类供我们使用,但有时我们可能需要创建自己的自定义异常类,以通过适当的消息和我们想要引入的任何自定义字段(如错误代码)通知调用者特定类型的异常。 。例如,假设我们编写了一个只处理文本文件的方法,因此当其他类型的文件作为输入发送时,我们可以为调用者提供适当的错误代码。
以下是自定义异常类的示例,并显示了它的用法。
package com.journaldev.exceptions;
public class MyException extends Exception {
private static final long serialVersionUID = 4664456874499611218L;
private String errorCode="Unknown_Exception";
public MyException(String message, String errorCode){
super(message);
this.errorCode=errorCode;
}
public String getErrorCode(){
return this.errorCode;
}
}
package com.journaldev.exceptions;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.IOException;
import java.io.InputStream;
public class CustomExceptionExample {
public static void main(String[] args) throws MyException {
try {
processFile("file.txt");
} catch (MyException e) {
processErrorCodes(e);
}
}
private static void processErrorCodes(MyException e) throws MyException {
switch(e.getErrorCode()){
case "BAD_FILE_TYPE":
System.out.println("Bad File Type, notify user");
throw e;
case "FILE_NOT_FOUND_EXCEPTION":
System.out.println("File Not Found, notify user");
throw e;
case "FILE_CLOSE_EXCEPTION":
System.out.println("File Close failed, just log it.");
break;
default:
System.out.println("Unknown exception occured, lets log it for further debugging."+e.getMessage());
e.printStackTrace();
}
}
private static void processFile(String file) throws MyException {
InputStream fis = null;
try {
fis = new FileInputStream(file);
} catch (FileNotFoundException e) {
throw new MyException(e.getMessage(),"FILE_NOT_FOUND_EXCEPTION");
}finally{
try {
if(fis !=null)fis.close();
} catch (IOException e) {
throw new MyException(e.getMessage(),"FILE_CLOSE_EXCEPTION");
}
}
}
}
请注意,我们可以有一个单独的方法来处理我们从不同方法获得的不同类型的错误代码,其中一些被消费,因为我们可能不希望通知用户,或者其中一些我们将返回以通知用户问题。
这里我扩展了Exception,这样每当生成这个异常时,它必须在方法中处理或返回给调用者程序,如果我们扩展RuntimeException,则不需要在throws子句中指定它。这是一个设计决策,但我总是喜欢检查异常,因为我知道在调用任何方法时我可以获得哪些异常并采取适当的操作来处理它们。
Exception in thread "main" java.lang.NullPointerException
at java.io.FileInputStream.<init>(FileInputStream.java:134)
at java.io.FileInputStream.<init>(FileInputStream.java:97)
at com.journaldev.exceptions.CustomExceptionExample.processFile(CustomExceptionExample.java:42)
at com.journaldev.exceptions.CustomExceptionExample.main(CustomExceptionExample.java:12)
在调试时,我们必须仔细查看堆栈跟踪以确定异常的实际位置。如果我们改变我们的实现逻辑,以便在下面检查这些异常;
private static void processFile(String file) throws MyException {
if(file == null) throw new MyException("File name can't be null", "NULL_FILE_NAME");
//further processing
}
然后,异常堆栈跟踪将如下所示,清楚地显示发生异常的位置以及清除消息。
com.journaldev.exceptions.MyException: File name can't be null
at com.journaldev.exceptions.CustomExceptionExample.processFile(CustomExceptionExample.java:37)
at com.journaldev.exceptions.CustomExceptionExample.main(CustomExceptionExample.java:12)
@throws
引发的异常 - 使用javadoc 清楚地指定方法抛出的异常,当您提供要使用的其他应用程序的接口时,它非常有用。这就是java中的异常处理,我希望你喜欢它并从中学到一些东西。