天天看点

Error 与 Exception 区别Error 与 Exception 区别

文章目录

  • Error 与 Exception 区别
    • Error
    • Exception
    • 常见面试题
    • 参考

Error 与 Exception 区别

Exception 和 Error 都是继承了 Throwable 类,在 Java中 只有 Throwable 类型的实例才可以被抛出(throw)或者捕获(catch),它是异常处理机制的基本组成类型。

Error 与 Exception 区别Error 与 Exception 区别

Error

Error 是指在正常情况下,不大可能出现的情况,绝大部分的 Error 都会导致程序(比如JVM自身)处于非正常的、不可恢复状态。既然是非正常情况,所以不便于也不需要捕获,常见的比如

OutOfMemoryError

之类,都是 Error 的子类。

Exception

Exception 是程序正常运行中,可以预料的意外情况,可能并且应该被捕获,进行相应处理。Exception 又分为检查(checked)异常、非检查(unchecked)异常,检查异常在源代码里必须显式地进行捕获处理,这是编译期检查的一部分。非检查异常就是所谓的运行时异常,类似

NullPointerException

ArrayIndexOutOfBoundsException

之类,通常是可以编码避免的逻辑错误,具体根据需要来判断是否需要捕获,并不会在编译期强制要求。

RuntimeException

类的子类属于非检查异常,除此之外继承自 Exception 的类为检查异常。

  • 通过 throws 关键字声明抛出异常
private static void readFile(String filePath) throws IOException {
        File file = new File(filePath);
        String result;
        BufferedReader reader = new BufferedReader(new FileReader(file));
        while ((result = reader.readLine()) != null) {
            System.out.println(result);
        }
        reader.close();
    }
           
  • try-catch 处理异常
private static void readFile(String filePath) {
        File file = new File(filePath);
        String result;
        BufferedReader reader;
        try {
            reader = new BufferedReader(new FileReader(file));
            while ((result = reader.readLine()) != null) {
                System.out.println(result);
            }
            reader.close();
        } catch (IOException e) {
            e.printStackTrace();
        }
    }
           

常见面试题

  1. Error 和 Exception 区别是什么?

Error

错误通常为虚拟机相关错误,如系统崩溃,内存不足,堆栈溢出等,编译器不会对这类错误进行检测,JAVA 应用程序也不应对这类错误进行捕获,一旦这类错误发生,通常应用程序会被终止,仅靠应用程序本身无法恢复。

Exception

异常是可以在应用程序中进行捕获并处理的,通常遇到这种错误,应对其进行处理,使应用程序可以继续正常运行。

  1. 运行时异常和一般异常区别是什么?

编译器不会对运行时异常进行检测,没有 try-catch,方法签名中也没有 throws 关键字声明,编译依然可以通过。如果出现了 RuntimeException, 那一定是程序员的错误。

一般异常如果没有 try-catch,且方法签名中也没有用 throws 关键字声明可能抛出的异常,则编译无法通过。这类异常通常为应用环境中的错误,即外部错误,非应用程序本身错误,如文件找不到等。

  1. NoClassDefFoundError 和 ClassNotFoundException 区别?

NoClassDefFoundError

是一个 Error 错误,产生原因:如果 JVM 或者 ClassLoader 实例尝试加载类的时候找不到类的定义。要查找的类在编译的时候是存在的,运行的时候却找不到了,就会导致 NoClassDefFoundError。造成该问题的原因可能是打包过程漏掉了部分类,或者 jar 包出现损坏或者篡改。解决办法是查找那些在开发期间存在于类路径下但在运行期间却不在类路径下的类。

ClassNOtFoundException

是一个 Exception 属于检查异常,产生原因:Java 支持使用 Class.forName 方法来动态地加载类,任意一个类的类名如果被作为参数传递给这个方法都将导致该类被加载到 JVM 内存中,如果这个类在类路径中没有被找到,就会在运行时抛出 ClassNotFoundException 异常。解决该问题需要确保所需的类连同它依赖的包存在于类路径中,常见问题在于类名书写错误。另外还有一个原因就是:当一个类已经由某个类加载器加载到内存中了,此时另一个类加载器又尝试着动态地从同一个包中加载这个类。通过控制动态类加载过程,可以避免上述情况发生。

  1. JVM 是如何处理异常的?

在一个方法中如果发生异常,这个方法会创建一个异常对象,并转交给 JVM,该异常对象包含异常名称,异常描述以及异常发生时应用程序的状态。创建异常对象并转交给 JVM 的过程称为抛出异常。可能有一系列的方法调用,最终才进入抛出异常的方法,这一系列方法调用的有序列表叫做调用栈。

JVM 会顺着调用栈去查找看是否有可以处理异常的代码,如果有,则调用异常处理代码。当 JVM 发现可以处理异常的代码时,会把发生的异常传递给它。如果 JVM 没有找到可以处理该异常的代码块,JVM 就会将该异常转交给默认的异常处理器(默认处理器为 JVM 的一部分),默认异常处理器打印出异常信息并终止应用程序。

  1. throw 和 throws 的区别是什么?

throw

关键字用来抛出方法或代码块中的异常,检查异常和非检查异常都可以被抛出。

throws

关键字用在方法签名处,用来标识该方法可能抛出的异常列表。一个方法用 throws 标识了可能抛出的异常列表,调用该方法的方法中必须包含可处理异常的代码,否则也要在方法签名中用 throws 关键字声明相应的异常。

  1. 常见的 RuntimeException 有哪些?
  • ClassCastException

    (类转换异常)
  • IndexOutOfBoundsException

    (数组越界)
  • NullPointerException

    (空指针异常)

参考

[1] 极客时间:杨晓峰-java核心技术36讲

[2] 一文读懂 JAVA 异常处理