Java异常详解,面试再也不怕被问到!

createh53个月前 (02-01)技术教程24

01Java异常简介

1.1 什么是异常

程序运行时,发生的不被期望的事件,它阻止了程序按照程序员的预期正常执行,这就是异常。异常发生时,是任程序自生自灭,立刻退出终止。在Java中即,Java在编译或运行或者运行过程中出现的错误。

1.2 异常处理机制

异常处理机制能让程序在异常发生时,按照代码的预先设定的异常处理逻辑,针对性地处理异常,让程序尽最大可能恢复正常并继续执行,且保持代码的清晰。

Java中的异常可以是函数中的语句执行时引发的,也可以是程序员通过throw 语句手动抛出的,只要在Java程序中产生了异常,就会用一个对应类型的异常对象来封装异常,JRE就会试图寻找异常处理程序来处理异常。

Java异常机制用到的几个关键字:try、catch、finally、throw、throws。

1. try

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

2. catch

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

3. finally

-- finally语句块总是会被执行。它主要用于回收在try块里打开的物力资源(如数据库连接、网络连接和磁盘文件)。只有finally块,执行完成之后,才会回来执行try或者catch块中的return或者throw语句,如果finally中使用了return或者throw等终止方法的语句,则就不会跳回执行,直接停止。

4. throw

-- 用于抛出异常。

5. throws

-- 用在方法签名中,用于声明该方法可能抛出的异常。主方法上也可以使用throws抛出。如果在主方法上使用了throws抛出,就表示在主方法里面可以不用强制性进行异常处理,如果出现了异常,就交给JVM进行默认处理,则此时会导致程序中断执行。

1.3 产生异常的原因:

1. 用户输入了非法数据。

2. 要打开的文件不存在。

3. 网络通信时连接中断,或者JVM内存溢出。

这些异常有的是因为用户错误引起,有的是程序错误引起的,还有其它一些是因为物理错误引起的。

02Java异常的分类

2.1 java的异常分类如下:

1. Error:JVM内部的错误,程序员不能处理的。

2. Exception:指的是程序运行中产生的异常,用户可以使用处理格式处理。

3. 非受检异常(RuntimeException):指的是运行时异常,编译器不强制要求处理。

4. 受检异常(IOException):异常编译器强制要求处理。

2.2 常见非受检异常:

1. ClassCastException: 当试图将对象强制转换为不是实例的子类时,抛出该异常。

2. ArithmeticException: 当出现异常的运算条件时,抛出此异常。例如,一个整数"除以零"时,抛出此类的一个实例。

3. IllegalArgumentException:抛出的异常表明向方法传递了一个不合法或不正确的参数。

4. ArrayIndexOutOfBoundsException:用非法索引访问数组时抛出的异常。如果索引为负或大于等于数组大小,则该索引为非法索引。

5. IndexOutOfBoundsException: 指示某排序索引(例如对数组、字符串或向量的排序)超出范围时抛出。

6. NullPointerException: 当应用程序试图在需要对象的地方使用 null 时,抛出该异常。

7. NumberFormatException: 当应用程序试图将字符串转换成一种数值类型,但该字符串不能转换为适当格式时,抛出该异常。

8. StringIndexOutOfBoundsException: 此异常由 String 方法抛出,指示索引或者为负,或者超出字符串的大小。

9. UnsupportedOperationException: 当不支持请求的操作时,抛出该异常。

10. ClassNotFoundException 应用程序试图加载类时,找不到相应的类,抛出该异常。

11. IllegalAccessException 拒绝访问一个类的时候,抛出该异常。

12. InterruptedException 一个线程被另一个线程中断,抛出该异常。

13. NoSuchMethodException 请求的方法不存在。

2.3 常见受检异常:

1. FileNotFoundException: 文件没有找到

2. IOException:io流异常

3. SQLException:sql异常

2.4 异常常用方法

下面是 Throwable 类的主要方法:

1. public String getMessage():返回关于发生的异常的详细信息。这个消息在Throwable 类的构造函数中初始化了。

2. public Throwable getCause():返回一个Throwable 对象代表异常原因。

3. public String toString():使用getMessage()的结果返回类的串级名字。

4. public void printStackTrace():打印toString()结果和栈层次到System.err,即错误输出流。

5. public StackTraceElement [] getStackTrace():返回一个包含堆栈层次的数组。下标为0的元素代表栈顶,最后一个元素代表方法调用堆栈的栈底。

6. public Throwable fillInStackTrace():用当前的调用栈层次填充Throwable 对象栈层次,添加到栈层次任何先前信息中。

03异常的使用及执行流程

3.1 异常的处理方案

try...catch、try...catch...finally、try...finally

try{

可能会发生的异常

}catch(异常类型 异常名(变量)){

针对异常进行处理的代码

}catch(异常类型 异常名(变量)){

针对异常进行处理的代码

}...

finally{

释放资源代码;

}


注意:

1. catch 不能独立于 try 存在。

2. catch里面不能没有内容。

3. 在 try/catch 后面添加 finally 块并非强制性要求的。

4. try 代码后不能既没 catch 块也没 finally 块。try里面越少越好。

5. try, catch, finally 块之间不能添加任何代码。

6. finally里面的代码最终一定会执行(除了JVM退出)。

7. 如果程序可能存在多个异常,需要多个catch进行捕获。

8. 异常如果是同级关系,catch谁前谁后没有关系,如果异常之间存在上下级关系,上级需要放在后面。

3.2 异常的执行流程

异常是一起处理好还是分开处理好?

根据实际的开发要求是否严格来决定。在实际的项目开发项目工作中,所有的异常是统一使用Exception处理还是分开处理,完全根据开发者的项目开发标准来决定。如果项目开发环境严谨,基本上要求针对每一种异常分别进行处理,并且要详细记录下异常产生的时间以及产生的位置,这样可以方便程序维护人员进行代码的维护。再次注意:处理多个异常时,捕获范围小的异常要放在捕获范围大的异常之前处理。

3.3 throw和throws的区别

throw和throws都是在异常处理中使用的关键字,区别如下:

throw:指的是在方法中人为抛出一个异常对象(这个异常对象可能是自己实例化或者抛出已存在的);

throws:在方法的声明上使用,表示此方法在调用时必须处理异常。

3.4 try-catch和throws开发中如何选择

1. 在实际开发过程中尽量将异常抛到controller层进行捕获,避免事务失效等问题。

2. 不涉及事务,异常了但不能影响程序向下执行,可以在当前位置进行捕获。

3.5 finally块和return

首先一个不容易理解的事实:在 try块中即便有return,break,continue等改变执行流的语句,finally也会执行。

finally中的return 会覆盖 try 或者catch中的返回值。

finally中的return或异常会抑制(消灭)前面try或者catch块中的异常。

3.6 assert关键字(了解)

在Java中,assert关键字是从JAVA SE 1.4 引入的,为了避免和老版本的Java代码中使用了assert关键字导致错误,Java在执行的时候默认是不启动断言检查的(这个时候,所有的断言语句都 将忽略!),如果要开启断言检查,则需要用开关-enableassertions或-ea来开启。

assert关键字语法很简单,有两种用法:

1. assert <boolean表达式>

如果<boolean表达式>为true,则程序继续执行。如果为false,则程序抛出AssertionError,并终止执行。

2. assert <boolean表达式> : <错误信息表达式>

如果<boolean表达式>为true,则程序继续执行。如果为false,则程序抛出java.lang.AssertionError,并输入<错误信息表达式>。

例如:

public class Test {

public static void main(String[] args) {

int a = 10;

int b = 2;

assert a == 10:"a不等于10";

System.out.println("a="+a);

}

}

执行结果为:a=10

public class Test {

public static void main(String[] args) {

int a = 10;

int b = 2;

assert a == 20:"a不等于20";

System.out.println("a="+a);

}

}

执行结果为:java.lang.AssertionError: a不等于20

04自定义异常

在 Java 中你可以自定义异常。如果要自定义异常类,则扩展Exception类即可,因此这样的自定义异常都属于检查异常(checked exception)。如果要自定义非检查异常,则扩展自RuntimeException。

按照国际惯例,自定义的异常应该总是包含如下的构造函数:

一个无参构造函数

一个带有String参数的构造函数,并传递给父类的构造函数。

一个带有String参数和Throwable参数,并都传递给父类构造函数

一个带有Throwable 参数的构造函数,并传递给父类的构造函数。

下面是IOException类的完整源代码,可以借鉴。

public class IOException extends Exception {

static final long serialVersionUID = 7818375828146090155L;

public IOException() {

super();

}

public IOException(String message) {

super(message);

}

public IOException(String message, Throwable cause) {

super(message, cause);

}

public IOException(Throwable cause) {

super(cause);

}

}

05总结

通过这篇文章,我们需要掌握的是什么是异常,异常的分类,异常的处理和捕获和自定义异常,通常开发中为了避免抛异常而终止,会进行捕获,给一个友好提示,异常机制也是为了让我们的业务更清晰,程序更加健壮,对用户更加友好。

相关文章

Java 异常处理通关指南(java异常处理是怎样实现的)

前言在理想世界中,程序永远不会出现问题,用户输入的数据永远是正确的,逻辑没有任何问题 ,选择打开的文件也一定是存在的,内存永远是够用的……!但是现实世界里一旦出现这些问题,如果处理不好,程序就不能正常...

5分钟课堂:Java异常处理(java异常处理方法及流程)

异常是在程序执行过程中发生的意外情况或错误。它表示程序在运行时出现了超出正常流程的状况。Java中的异常处理机制用来处理程序运行时发生的各种非正常情况(即异常)。异常机制优点使用异常机制分离了代码中的...

异常处理,JAVA中异常处理的介绍(java中异常处理机制是怎样的)

异常处理的介绍在Java程序的运行过程中,如果Java虚拟机检测出一个无法执行的操作,就会产生运行时错误(runtime error)在Java中,运行时错误会作为异常来抛出。抛出的异常是一个对象,该...

JAVA中异常,你都理解了吗?(java异常处理规则(新手必看))

人非圣贤,孰能无过,何况语言。JAVA有成熟的异常处理机制,正确使用它可以帮我们解决程序的健壮性。1.Java中的异常机制java异常类图:Error一般不会出现,大多数非正常的情况,比如内存溢出等。...

关于Java Exception异常的深入用法及实例

关于Java Exception异常的深入用法及实例一、异常的概念生活中的异常:例如感冒发烧,工作时电脑蓝屏、死机等。程序中的异常:在程序运行的过程中,也会发生这种非正常状况,例如程序运行时磁盘空间不...

学习Java快速入门全流程体系介绍-异常处理

Java异常处理是指在程序运行时检测到异常情况并尝试以适当的方式进行处理的机制。Java中提供了一种try-catch-finally结构来处理异常。处理异常的主要目的是使程序在遇到错误时能够优雅地退...