来了!令人头疼的 Java 异常面试总结

简介: 来了!令人头疼的 Java 异常面试总结

异常简介

异常类层次结构

image.png从结构图可以看出,所有异常均继承自 Throwable 类,它有两个重要的子类:Exception 和 Error ,各自又包含大量子类。


Exception

程序本身可以处理的异常,又可以分为 受检异常 和 非受检异常 ,受检异常 可以用 try...catch... 语句进行捕获处理,而且能从异常中恢复。但 非受检异常 是程序运行时错误,会导致程序崩溃而无法恢复。


受检异常

编译器要求必须处理的异常,正确的程序在运行时,经常会出现、预期范围内的情况。一旦发生该类异常,就必须使用某种方式进行处理。包括除开 RuntimeException 及其子类之外的 Exception 异常。编译器会检查此类异常,所以我们必须使用 throws 进行抛出或者 try...catch 进行捕获,否则将导致编译失败。


非受检异常

编译器不会检查而且也不要求我们进行处理,即就算在程序中出现了此类异常,即便我们没有用 try...catch 进行捕获或者用 throws 进行抛出,编译都会成功。包括 RuntimeException 及其子类和错误 Error.


同时也可以分为:运行时异常和编译时异常。


运行时异常

RuntimeException 类及其子类,表示 JVM 在运行期间可能出现的异常,Java 编译器不会检查它。没有通过 throws 抛出或 try...catch 捕获,仍然可以编译通过,常见的有 NullPointerException、ArrayIndexOutBoundException、ClassCastException、ArithmeticException、NumberFormatException、IllegalArgumentException;


编译时异常

Exception 中除开运行时异常之外的异常,Java 编译器会检查它,一旦出现,必须使用 throws 进行声明抛出,或者使用 try...catch 进行捕获异常,否则不能通过编译。常见的有 ClassNotFoundException、IOException。在程序中,通常不会自定义该类异常,而是直接用系统提供的异常类,该异常必须手动在代码中添加捕获语句来处理。


Error

程序无法处理的错误,表示程序运行过程中教严重的问题,大多与 coder 所做操作无关,而是代码运行时 JVM 出现的问题。此时说明故障发生于虚拟机本身、或者发生在虚拟机试图执行应用时。


Throwable 常用方法

方法 说明

public String getMessage() 返回异常发生时的简要描述

public String toString() 返回异常发生时的详细信息

public String getLocalizeMessage() 返回异常对象的本地化信息,若子类重写该方法,可以生成本地化信息,若未重写,则返回信息同 getMessage() 方法

public void printStackTrace() 在控制台中打印异常对象封装的异常信息

try-catch-finally 和 try-with-resources

try-catch-finally

try :用于捕获异常,后接零个或多个 catch,没有 catch 则必须加上 finally;

catch:用于处理 try 捕获到的异常;

finally:无论是否捕获/处理异常,finally 块中内容均会执行,就算 try 或 catch 中有 return 语句,finally 中代码也将在方法返回之前执行;

try-with-resources

当我们有必须要关闭的资源时,建议优先使用 try-with-resources,这样写出的代码更加简短清晰。


两者对比


// try-catch-finally
Scanner scanner = null;
try {
    scanner = new Scanner(new File("D:/demo.txt"));
    while (scanner.hasNext()) {
        System.out.println(scanner.nextLine());
    }
} catch (FileNotFoundException e) {
    e.printStackTrace();
} finally {
    if (scanner != null) {
        scanner.close();
    }
}
// try-with-resources
try (Scanner scanner = new Scanner(new File("D:/demo.txt"))) {
    while (scanner.hasNext()) {
        System.out.println(scanner.nextLine());
    }
} catch (FileNotFoundException e) {
    e.printStackTrace();
}

异常处理

Java 中,异常处理机制分为 声明异常、抛出异常和捕获异常,根据异常的情况,可以对异常进行不同处理:image.png声明异常

对于知道如何进行处理的异常,一般要进行捕获,但此时不知道如何将处理的异常继续传递下去,可以通过在方法签名中使用 throws 来声明可能抛出的异常,有如下两点需要注意:


非受检异常(Error、RuntimeException 及其子类) 不能使用 throws 关键字来声明要抛出的异常;

一个方法出现编译时异常,就需要 try...catch/throws 进行处理,否则会导致编译失败 ;

抛出异常

一旦觉得某些异常无法处理,但同时又不用我们进行处理,那我们就可以将其抛出。一般是使用 throw 在方法内部抛出一个 Throwable 类型的异常。


捕获异常

程序在运行前一般不会报错,但是运行后可能出现某些未知错误,如果不想直接抛出给上一级处理,那我们就需要通过 try...catch... 的形式对异常进行捕获,然后根据不同的情况来进行相应处理。


异常常见面试题

Error 和 Exception 的区别?

Exception 类的异常能够在程序中进行捕获并处理,遇到该类异常,应该进行处理,从而使程序能够继续正常运行;


Error 类的错误一般是虚拟机相关错误,如系统崩溃、内存不足、堆栈溢出等,编译器不会检测这类错误。我们也不会对这类错误进行捕获,一旦发生,一般都会导致程序崩溃无法恢复;


运行时异常和受检异常的区别?

运行时异常包括 RuntimeException 及其子类,表示 JVM 运行期间可能出现的异常,不会被 Java 编译器检查。


而受检异常是除开 RuntimeException 及其子类之外的其他 Exception,会被 Java 编译器检查。


两者的 区别 在于:是否需要调用者必须处理该异常,如果必须处理,则一般使用受检异常,否则一般选择非受检异常(RuntimeException);


throw 和 throws 的区别?

throw:用于在方法内部抛出异常对象

throw 用在方法体内,表示抛出异常,由方法体内的语句处理;

throw 是具体向外抛出异常的动作,所以抛出的是一个异常实例,执行 throw 一定是抛出了某种异常;

throws:用于在方法签名上声明该方法所要抛出的异常

throws 语句使用在方法声明后,表示若抛出异常,则由该方法的调用者来进行异常的处理;

throws 主要是声明这个方法会抛出某种类型的异常,让它的使用者要知道需要捕获的异常的类型;

throws 表示出现异常的一种可能性,并非一定发生该种异常;

final、finally、finallize 的区别?

final 用于修饰类、方法、变量,修饰类时表示类不能被继承;修饰方法时表示方法不能别重写,但是能够被重载;修饰变量时表示该变量是一个常量无法被重写赋值;


finally 一般作用于 try...catch 代码块,处理异常时,通常将必须要执行的代码放在 finally 代码块中,表示无论是否出现异常,此代码块均执行,一般用来存放一些关闭资源的代码;


finallize 是一个方法,属于 Object 类,Java 允许用 finallize() 方法在垃圾回收器将对象从内存中清除前做一些必要的清理工作;


常见的 RuntimeException 异常?

ClassCastException

IndexOutOfBoundsException

NullPointerException

ArrayStoreException

BufferOverFlowException

JVM 如何处理异常?

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


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


目录
相关文章
|
5天前
|
负载均衡 NoSQL 算法
一天五道Java面试题----第十天(简述Redis事务实现--------->负载均衡算法、类型)
这篇文章是关于Java面试中Redis相关问题的笔记,包括Redis事务实现、集群方案、主从复制原理、CAP和BASE理论以及负载均衡算法和类型。
一天五道Java面试题----第十天(简述Redis事务实现--------->负载均衡算法、类型)
|
5天前
|
安全 Java 数据库
一天十道Java面试题----第四天(线程池复用的原理------>spring事务的实现方式原理以及隔离级别)
这篇文章是关于Java面试题的笔记,涵盖了线程池复用原理、Spring框架基础、AOP和IOC概念、Bean生命周期和作用域、单例Bean的线程安全性、Spring中使用的设计模式、以及Spring事务的实现方式和隔离级别等知识点。
|
6天前
|
存储 监控 安全
一天十道Java面试题----第三天(对线程安全的理解------>线程池中阻塞队列的作用)
这篇文章是Java面试第三天的笔记,讨论了线程安全、Thread与Runnable的区别、守护线程、ThreadLocal原理及内存泄漏问题、并发并行串行的概念、并发三大特性、线程池的使用原因和解释、线程池处理流程,以及线程池中阻塞队列的作用和设计考虑。
|
4天前
|
存储 缓存 网络协议
复盘女朋友面试4个月的Java基础题
这篇文章是关于Java基础面试题的复盘,涵盖了HashMap原理、对象序列化作用等高频面试问题,并强调了Java基础知识的重要性。
复盘女朋友面试4个月的Java基础题
|
5天前
|
存储 NoSQL Java
一天五道Java面试题----第十一天(分布式架构下,Session共享有什么方案--------->分布式事务解决方案)
这篇文章是关于Java面试中的分布式架构问题的笔记,包括分布式架构下的Session共享方案、RPC和RMI的理解、分布式ID生成方案、分布式锁解决方案以及分布式事务解决方案。
一天五道Java面试题----第十一天(分布式架构下,Session共享有什么方案--------->分布式事务解决方案)
|
5天前
|
存储 安全 Java
一天十道Java面试题----第二天(HashMap和hashTable的区别--------》sleep、wait、join)
这篇文章是关于Java面试的第二天笔记,涵盖了HashMap与HashTable的区别、ConcurrentHashMap的实现原理、IOC容器的实现方法、字节码的概念和作用、Java类加载器的类型、双亲委派模型、Java异常体系、GC如何判断对象可回收、线程的生命周期及状态,以及sleep、wait、join、yield的区别等十道面试题。
一天十道Java面试题----第二天(HashMap和hashTable的区别--------》sleep、wait、join)
|
4天前
|
存储 Java 程序员
|
5天前
|
SQL Java 数据库连接
一天五道Java面试题----第六天(1)
这篇文章是关于Java面试中常见的五个问题,包括MyBatis和Hibernate的对比、MyBatis中#{}和${}的区别、MyBatis插件的运行原理及开发流程、索引的基本原理以及MySQL聚簇索引和非聚簇索引的区别。
|
5天前
|
缓存 NoSQL Redis
一天五道Java面试题----第九天(简述MySQL中索引类型对数据库的性能的影响--------->缓存雪崩、缓存穿透、缓存击穿)
这篇文章是关于Java面试中可能会遇到的五个问题,包括MySQL索引类型及其对数据库性能的影响、Redis的RDB和AOF持久化机制、Redis的过期键删除策略、Redis的单线程模型为何高效,以及缓存雪崩、缓存穿透和缓存击穿的概念及其解决方案。
|
5天前
|
前端开发 Java 数据库连接
一天十道Java面试题----第五天(spring的事务传播机制------>mybatis的优缺点)
这篇文章总结了Java面试中的十个问题,包括Spring事务传播机制、Spring事务失效条件、Bean自动装配方式、Spring、Spring MVC和Spring Boot的区别、Spring MVC的工作流程和主要组件、Spring Boot的自动配置原理和Starter概念、嵌入式服务器的使用原因,以及MyBatis的优缺点。