Java 异常面试题总结(2022最新版)

关于作者
🐶 程序猿周周
⌨️ 短视频小厂BUG攻城狮
🤺 如果文章对你有帮助,记得关注、点赞、收藏,一键三连哦,你的支持将成为我最大的动力

本文是《后端面试小册子》系列的第 1️⃣1️⃣ 篇文章,该系列将整理和梳理笔者作为 Java 后端程序猿在日常工作以及面试中遇到的实际问题,通过这些问题的系统学习,也帮助笔者顺利拿到阿里、字节、华为、快手等多个大厂 Offer,也祝愿大家能够早日斩获自己心仪的 Offer。


PS:《后端面试小册子》已整理成册,目前共十三章节,总计约二十万字,欢迎👏🏻关注公众号【程序猿周周】获取电子版和更多学习资料(最新系列文章也会在此陆续更新)。公众号后台可以回复关键词「电⼦书」可获得这份面试小册子。文中所有内容都会在 Github 开源,项目地址 csnotes,如文中存在错误,欢迎指出。如果觉得文章还对你有所帮助,赶紧点个免费的 star 支持一下吧!

在这里插入图片描述

标题地址
MySQL数据库面试题总结(2022版)https://blog.csdn.net/adminpd/article/details/122910606
Redis面试题总结(2022版)https://blog.csdn.net/adminpd/article/details/122934938
计算机网络面试题总结(2022版)https://blog.csdn.net/adminpd/article/details/122973684
操作系统面试题总结(2022版)https://blog.csdn.net/adminpd/article/details/122994599
Linux面试题总结(2022版)https://blog.csdn.net/adminpd/article/details/122994862
Spring面试题总结(2022版)https://blog.csdn.net/adminpd/article/details/123016872
Java基础面试题总结(2022版)https://blog.csdn.net/adminpd/article/details/123080189
Java集合面试题总结(2022版)https://blog.csdn.net/adminpd/article/details/123171501
Java并发面试题总结(2022版)https://blog.csdn.net/adminpd/article/details/123266624
Java虚拟机面试题总结(2022版)https://blog.csdn.net/adminpd/article/details/123412605
Java异常面试题总结(2022版)https://blog.csdn.net/adminpd/article/details/123462676
设计模式面试题总结(2022版)https://blog.csdn.net/adminpd/article/details/123490442
Dubbo面试题总结(2022版)https://blog.csdn.net/adminpd/article/details/123538243
Netty面试题总结(2022版)https://blog.csdn.net/adminpd/article/details/123564362

1 体系结构

1、什么事 Java 的异常机制?

Java 异常是 Java 提供的一种识别及响应错误的一致性机制
Java 异常机制可以使程序中异常处理代码和正常业务代码分离,保证程序代码更加优雅,并提高程序健壮性。在有效使用异常的情况下,异常能清晰的回答What, Where, Why 这3个问题:异常类型回答了“什么”被抛出,异常堆栈跟踪回答了“在哪”抛出,异常信息回答了“为什么”会抛出。

2、Java 异常的体系结构?

Java 异常机制中定义了许多异常类,并且定义了基类 java.lang.Throwable 作为所有异常的超类。Java 语言设计者将异常划分为两类:Error 和 Exception,其体系结构大致如下图所示:

在这里插入图片描述

2.1 Error

Error 是程序中无法处理的错误,表示运行应用程序中出现了严重的错误。此类错误一般表示代码运行时 JVM 出现问题。通常有 Virtual MachineError(虚拟机运行错误)NoClassDefFoundError(类定义错误)OutOfMemoryError(内存不足错误)StackOverflowError(栈溢出错误) 等。

此类错误发生时,JVM 将终止线程。同时这些错误是不可查的,非代码性错误。因此,当此类错误发生时,应用不应该去处理此类错误,也无法处理。

2.2 Exception

Exception 是程序本身可以捕获并且可以处理的异常。这种异常又分为两类:运行时异常编译异常

1)运行时异常

也称Unchecked(不受检查异常),这类异常包括 RuntimeException 类及其子类,表示 JVM 在运行期间可能出现的异常。

比如说试图使用空值对象的引用(NullPointerException)、数组下标越界(ArrayIndexOutBoundException)。此类异常属于不可查异常,一般是由程序逻辑错误引起的,在程序中可以选择捕获处理,也可以不处理。

2)编译异常

这类异常包括 Exception 中除 RuntimeException 及其子类之外的异常。又称Checked(受检查异常)

如果程序中出现此类异常,比如 ClassNotFoundException(没有找到指定的类异常)IOException(IO流异常)等,要么通过 throws 语句进行声明抛出,要么通过 try-catch 语句捕获处理,否则不能通过编译。

在程序中,通常不会自定义该类异常,而是直接使用系统提供的异常类。该异常我们必须手动在代码里添加捕获语句来处理该异常。

3、什么是受检异常与非受检异常?

Java 的所有异常可以分为受检异常(Checked Exception)和非受检异常(Unchecked Exception)。

  • 受检查异常

编译器要求必须处理的异常。正确的程序在运行过程中,经常容易出现的、符合预期的异常情况。一旦发生此类异常,就必须采用某种方式进行处理。除 RuntimeException 及其子类外,其他的 Exception 异常都属于受检异常。编译器会检查此类异常,也就是说当编译器检查到应用中的某处可能会此类异常时,将会提示你处理本异常——要么使用try-catch捕获,要么使用方法签名中用 throws 关键字抛出,否则编译不通过。

  • 非受检查异常

编译器不会进行检查并且不要求必须处理的异常,也就说当程序中出现此类异常时,即使我们没有 try-catch 捕获它,也没有使用 throws 抛出该异常,编译也会正常通过。该类异常包括运行时异常(RuntimeException 及其子类)和错误(Error)

2 异常使用

1、异常关键字以及含义?

  • try :用于监听。

将要被监听代码,即可能抛出异常的代码放在 try 语句块之内,当 try 语句块内发生异常时,异常就被抛出。

  • catch:用于捕获异常。

catch 用来捕获 try 语句块中发生的异常。

一个 try 语句后可以跟多个 catch 语句以捕获多个异常类型,并对不同类型的异常做出不同的处理。而同一个 catch 也可以用 | 隔开捕获多种类型异常。

  • finally:总是最后被执行。

它主要用于回收在 try 块里打开的物力资源(如数据库连接、网络连接和磁盘文件)。

只有 finally 块中语句执行完成之后,才会回来执行 try 或者 catch 块中的 return 或 throw 语句,如果 finally 中使用了 return 或 throw 等终止方法的语句,则就不会跳回执行,直接停止。

  • throw:用于抛出异常。
  • throws:用在方法签名中,用于声明该方法可能抛出的异常。

2、Finally 不会被执行的情况?

  • 在前面的代码中用了System.exit()退出程序。
  • finally 语句块中发生了异常。
  • 程序所在的线程死亡。

3、什么是 try-with-resource?

try {
    // ...
} catch (IOException e) {
    // ...
} finally {
    if (null != reader) {
        try {
            reader.close();
        } catch (IOException e) {
            e.printStackTrace();
        }
    }
}

上述代码中,finally 块的 close 方法也可能抛出 IOException, 从而覆盖了原始异常。JAVA 7 提供了更优雅的方式来实现资源的自动释放,但需要自动释放的资源需要实现 CloseableAutoCloseable 接口。

同时还需将要关闭的外部资源在 try() 中创建,catch()捕获处理异常。其实 try-with-resource 机制是一种语法糖,其底层实现原理仍然是 try-catch-finally 写法,不过在 catch 代码块中加入一个 addSuppressed() 方法,即异常抑制方法。如果业务处理和关闭连接都出现了异常,业务处理的异常会抑制关闭连接的异常,只抛出处理中的异常,仍然可以通过 getSuppressed() 方法获得关闭连接的异常。

try(FileInputStream fis = new FileInputStream(new File("./test.txt"))){
    // do something
}catch(IOExpection e){
    // ...
}

4、Java 是如何处理异常的?

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

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

5、throw 和 throws 的区别?

  • 作用位置

throw 关键字用在方法内部,而 throws 关键字用在方法声明上。

  • 异常个数

throw 只能抛出一种异常,用来抛出方法或代码块中的异常,受查异常和非受查异常都可以被抛出。
throws 可以抛出多个异常,用来标识该方法可能抛出的异常列表。

6、NoClassDefFoundError 和 ClassNotFoundException 区别?

  • NoClassDefFoundError 是一个 Error 类型的异常,是由 JVM 引起的,不应该尝试捕获这个异常。

引起该异常的原因是 JVM 或 ClassLoader 尝试加载某类时在内存中找不到该类的定义,该动作发生在运行期间,即编译时该类存在,但是在运行时却找不到了,可能是变异后被删除了等原因导致。

  • ClassNotFoundException 是一个受查异常,需要显式地使用 try-catch 对其进行捕获和处理,或在方法签名中用 throws 关键字进行声明。

当使用 Class.forName、ClassLoader.loadClass 或 ClassLoader.findSystemClass 动态加载类到内存的时候,通过传入的类路径参数没有找到该类,就会抛出该异常;另一种抛出该异常的可能原因是某个类已经由一个类加载器加载至内存中,另一个加载器又尝试去加载它。

7、catch 是否可以捕获子类异常?

try {
	throw new ExampleB("Example B")
} catchExampleA e){
	System.out.println("Example A");
} catchException e){
	System.out.println("Exception");
}

根据里式代换原则,能使用父类型的地方一定能使用子类型。故上面一段代码会输出 Example A,即 catch 可以捕获声明异常以及所有子类,这也是我们不推荐直接捕获 Exception 类的原因。

3、最佳实践

节选自《阿里巴巴Java开发手册》。

1)【强制】Java 类库中定义的可以通过预检查方式规避的 RuntimeException 异常不应该通过 catch 的方式来处理,比如 NullPointerException、IndexOutOfBoundsException 等。

说明:无法通过预检查的异常除外,比如,在解析字符串形式的数字时,可能存在数字格式错误,不得不通过catch NumberFormatException来实现。 正例:if (obj != null) {…} 反例:try { obj.method(); } catch (NullPointerException e) {…}

2)【强制】异常不要用来做流程控制,条件控制。

说明:异常设计的初衷是解决程序运行中的各种意外情况,且异常的处理效率比条件判断方式要低很多。

3)【强制】使用 catch 时请分清稳定代码和非稳定代码,稳定代码指的是无论如何不会出错的代码。对于非稳定代码的catch尽可能进行区分异常类型,再做对应的异常处理。

说明:对大段代码进行 try-catch,使程序无法根据不同的异常做出正确的应激反应,也不利于定位问题,这是一种不负责任的表现。 正例:用户注册的场景中,如果用户输入非法字符,或用户名称已存在,或用户输入密码过于简单,在程序上作出分门别类的判断,并提示给用户。

4)【强制】捕获异常是为了处理它,不要捕获了却什么都不处理而抛弃之,如果不想处理它,请将该异常抛给它的调用者。最外层的业务使用者,必须处理异常,将其转化为用户可以理解的内容。

5)【强制】有try块放到了事务代码中,catch异常后,如果需要回滚事务,一定要注意手动回滚事务。

6)【强制】finally 块必须对资源对象、流对象进行关闭,有异常也要做try-catch。 说明:如果JDK7及以上,可以使用try-with-resources方式。

7)【强制】不要在 finally 块中使用return。 说明:try块中的return语句执行成功后,并不马上返回,而是继续执行finally块中的语句,如果此处存在return语句,则在此直接返回,无情丢弃掉try块中的返回点。 反例:

private int x = 0;
public int checkReturn() {
    try {
        // x等于1,此处不返回
        return ++x;
    } finally {
        // 返回的结果是2
        return ++x;
    }
}

8)【强制】捕获异常与抛异常,必须是完全匹配,或者捕获异常是抛异常的父类。 说明:如果预期对方抛的是绣球,实际接到的是铅球,就会产生意外情况。

9)【强制】在调用 RPC、二方包、或动态生成类的相关方法时,捕捉异常必须使用 Throwable 类来进行拦截。

说明:通过反射机制来调用方法,如果找不到方法,抛出 NoSuchMethodException。什么情况会抛出 NoSuchMethodError 呢?二方包在类冲突时,仲裁机制可能导致引入非预期的版本使类的方法签名不匹配,或者在字节码修改框架(比如:ASM)动态创建或修改类时,修改了相应的方法签名。这些情况,即使代码编译期是正确的,但在代码运行期时,会抛出 NoSuchMethodError。

10)【推荐】方法的返回值可以为 null,不强制返回空集合,或者空对象等,必须添加注释充分说明什么情况下会返回 null 值。

说明:本手册明确防止 NPE 是调用者的责任。即使被调用方法返回空集合或者空对象,对调用者来说,也并非高枕无忧,必须考虑到远程调用失败、序列化失败、运行时异常等场景返回 null 的情况。

11)【推荐】防止 NPE 是程序员的基本修养,注意 NPE 产生的场景: 1) 返回类型为基本数据类型,return 包装数据类型的对象时,自动拆箱有可能产生 NPE。2) 数据库的查询结果可能为 null。3)集合里的元素即使 isNotEmpty,取出的数据元素也可能为 null。4)远程调用返回对象时,一律要求进行空指针判断,防止 NPE。5)对于 Session 中获取的数据,建议进行 NPE 检查,避免空指针。6)级联调用如 obj.getA().getB().getC(); 一连串调用,易产生NPE。

推荐使用 JDK8 的 Optional 类来防止 NPE 问题。

12)【推荐】定义时区分 unchecked / checked 异常,避免直接抛出 new RuntimeException(),更不允许抛出 Exception 或者 Throwable,应使用有业务含义的自定义异常。推荐业界已定义过的自定义异常,如:DAOException / ServiceException 等。

13)【参考】对于公司外的 http/api 开放接口必须使用”错误码”形式;而应用内部推荐异常抛出;跨应用间 RPC 调用优先考虑使用 Result 方式,封装 isSuccess() 方法、“错误码”、“错误简短信息”。

说明:关于 RPC 方法返回方式使用 Result 方式的理由: 1)使用抛异常返回方式,调用方如果没有捕获到就会产生运行时错误。 2)如果不加栈信息,只是自定义异常,加入自己的理解的 error message,对于调用端解决问题的帮助不会太多。如果加了栈信息,在频繁调用出错的情况下,数据序列化和传输的性能损耗也是问题。

14)【参考】避免出现重复的代码(Don’t Repeat Yourself),即 DRY 原则。

说明:随意复制和粘贴代码,必然会导致代码的重复,在以后需要修改时,需要修改所有的副本,容易遗漏。必要时抽取共性方法,或者抽象公共类,甚至是组件化。

  • 5
    点赞
  • 57
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 1
    评论
### 回答1: 您好,以下是Java基础面试题2022最新版汇总): 1. Java中的四种访问修饰符是什么?它们分别代表什么含义? 答:Java中的四种访问修饰符是public、protected、default和private。它们分别代表公共的、受保护的、默认的和私有的含义。 2. Java中的八种基本数据类型是什么? 答:Java中的八种基本数据类型是byte、short、int、long、float、double、char和boolean。 3. Java中的自动装箱和拆箱是什么? 答:Java中的自动装箱和拆箱是指将基本数据类型自动转换为对应的包装类类型,以及将包装类类型自动转换为对应的基本数据类型。 4. Java中的final关键字有哪些用途? 答:Java中的final关键字有三种用途:用于修饰变量,表示该变量的值不可改变;用于修饰方法,表示该方法不可被重写;用于修饰类,表示该类不可被继承。 5. Java中的抽象类和接口有什么区别? 答:Java中的抽象类和接口都不能被实例化,但是抽象类可以包含非抽象方法,而接口只能包含抽象方法。另外,一个类只能继承一个抽象类,但是可以实现多个接口。 6. Java中的多态是什么? 答:Java中的多态是指同一个方法可以根据不同的对象调用出不同的行为。它实现的方式有两种:重载和重写。 7. Java中的异常处理机制是什么? 答:Java中的异常处理机制是指通过try-catch-finally语句块来捕获和处理程序运行时可能出现的异常情况,保证程序的稳定性和可靠性。 8. Java中的线程是什么?如何创建线程? 答:Java中的线程是指程序执行的最小单位,它可以独立运行,也可以与其他线程共享资源。创建线程的方式有两种:继承Thread类和实现Runnable接口。 9. Java中的集合框架有哪些?它们之间有什么区别? 答:Java中的集合框架包括List、Set、Map和Queue等。它们之间的区别在于:List是有序的、可重复的集合;Set是无序的、不可重复的集合;Map是键值对的集合;Queue是先进先出的队列。 10. Java中的反射是什么?有什么作用? 答:Java中的反射是指在程序运行时动态地获取类的信息,并可以在运行时操作类的属性和方法。它的作用在于:可以实现动态代理、动态生成代码、动态加载类等功能。 ### 回答2: Java是一种广泛使用的编程语言,它的应用覆盖多个领域,包括机器学习、大数据、企业级应用等。而Java开发人员也是当前市场上非常火热的职业之一。为了进入一家Java开发企业或者获取Java开发岗位,需要通过Java基础面试。 下面我将介绍Java基础面试题,并提供一些答案和解释: 1. 什么是Java? Java是一种面向对象的编程语言,被广泛应用于开发各种应用程序,包括Web应用程序、移动应用程序、桌面应用程序等。 2. 什么是面向对象编程? 面向对象编程是一种编程范型,它将程序分解为对象,每个对象都有自己的数据和方法,对象之间可以相互交互。面向对象编程通常具有封装、继承和多态等特性。 3. Java应用程序的基本结构是什么? Java应用程序的基本结构包括类定义,变量定义,方法定义和语句执行。 4. 什么是Java虚拟机? Java虚拟机是Java平台的核心组件之一,它提供了一个运行Java程序的环境。Java虚拟机可以在不同的操作系统上运行Java程序,实现了跨平台运行的目的。 5. 什么是Java语言的基本数据类型? Java语言的基本数据类型包括byte、short、int、long、float、double、boolean和char。 6. 如何声明一个变量? 在Java中,声明一个变量需要指定变量的类型和名称。例如,int count = 10; 7. 什么是包? 包是在Java中管理类和接口的一种方式。包可以将相关的类和接口组织在一起,以便更好地管理代码。 8. 什么是Java中的异常异常是在程序执行期间出现的错误,它包含了错误信息、错误类型和错误的上下文。 9. 什么是Java中的多线程? 多线程是Java中一种并发处理的方法,它允许程序同时执行多个任务。Java中的线程可以通过继承Thread类或实现Runnable接口来创建。 10. Java中的继承关系是什么? 继承是Java中一种对象之间的关系,它允许子类继承父类的属性和方法。子类可以通过继承父类来扩展自己的功能。 以上是常见的Java基础面试题,希望能够对Java开发者提供帮助。在面试过程中,还需要我们展示自己的编程能力和实际应用经验。只有掌握了扎实的Java基础、熟悉常见的应用场景、具备实际经验并且具备坚实的编程能力,才能获得Java开发岗位。 ### 回答3: Java作为目前最热门的编程语言之一,在面试中也是一个重点考察的领域。下面对于Java基础面试题进行汇总和解答。 1. Java常见的数据类型有哪些? 答:Java常见的数据类型有基本数据类型和引用数据类型两种。基本数据类型包括byte、short、int、long、float、double、boolean、char;引用数据类型包括类、数组、接口。 2. int和Integer有什么区别? 答:int是Java中的一种基本数据类型,使用时直接声明并赋值即可;而Integer是int的包装类型,是一个对象,需要通过new来进行初始化,也可以使用自动装箱和拆箱进行转换。 3. String和StringBuilder的区别? 答:String是一个不可变的类,一旦创建就不能修改,修改需要重新创建对象;StringBuilder是一个可变的类,可以通过append、insert等方法修改字符串,节省内存和时间。 4. 抽象类和接口的区别? 答:抽象类和接口都是用于描述行为的概念,但是抽象类更偏向于在类层次结构中描述一种抽象的类,它可以拥有抽象方法和具体实现方法,可以有构造函数,但不能被实例化;而接口更偏向于描述一种能力,只能包含抽象方法、常量和默认方法,不能有构造函数,需要实现接口才能使用。 5. 静态和非静态的区别? 答:静态属性或方法属于类,而非静态属性或方法属于实例,也就是说同一个类属性或方法,静态的只有一份,非静态每次实例化时都有自己的一份;静态变量在类的加载时初始化,公用一份内存,一旦被修改,则所有的对象都会共享该值;非静态变量在每次实例化时初始化,属于独立变量。 6. final关键字的作用? 答:final关键字可以修饰类、方法和变量,表示不可修改、不可继承或不可重写。final修饰的变量是一个常量,一旦初始化后就不能修改;final修饰的方法不能被子类重写;final修饰的类不能被继承。 7. try-catch-finally语句的应用? 答:try-catch-finally语句用于捕获程序中的异常,在try块中尝试执行一段可能会出现异常的代码,如果出现异常则会转到catch块中处理异常,最后无论是否出现异常都会执行finally块中的语句。finally块一般用于释放资源或进行清理操作,如关闭数据库、文件等。 8. 在Java中如何实现多线程? 答:Java中实现多线程有两种方式:继承Thread类和实现Runnable接口。继承Thread类需要重写run方法,而实现Runnable接口需要实现run方法。另外,还可以使用线程池、同步锁、信号量、线程等待等方式实现多线程。 9. 快速判断一个字符串是否是数字? 答:可以使用正则表达式或Java自带的API进行判断。正则表达式可以使用\d+匹配数字,Java自带的API可以使用try-catch捕获NumberFormatException异常来判断。 10. Java中的反射是什么? 答:Java中的反射是指程序在运行时可以动态获取和操控类的信息和对象的属性或方法,包括类的名称、属性和方法。可以通过反射来获取类的所有信息,包括私有的成员变量和方法,也可以在运行时创建对象、调用方法等。但是由于反射操作需要消耗一定的资源,因此不适合在频繁执行的代码中使用。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

程序猿周周

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值