一文了解java异常机制

简介: 一文了解java异常机制1.异常的概述1.1什么是异常?异常:程序在运行过程中发生由于外部问题导致的程序异常事件,发生的异常会中断程序的运行。(在Java等面向对象的编程语言中)异常本身是一个对象,产生异常就是产生了一个异常对象。

一文了解java异常机制
1.异常的概述
1.1什么是异常?
异常:程序在运行过程中发生由于外部问题导致的程序异常事件,发生的异常会中断程序的运行。(在Java等面向对象的编程语言中)异常本身是一个对象,产生异常就是产生了一个异常对象。注意在java中异常不是错误,在下文的异常的分类中有解释。
举个生活中的栗子来说明异常吧,假如你平时是开车上班,一般情况下你都能按时到公司上班,但是今天遇到特殊情况,就是遇到公路施工了,遇到这种情况,如果你没有做任何处理,就有可能导致上班迟到。这种“路上施工” 就是程序中的异常了,他是外部问题导致的事件,并不是你本身出现的问题。(题外话:然而老板并不在意你是什么问题导致迟到的,反正你就是迟到了。。。。)

1.2 如何处理异常?
1.2.1 传统的异常处理
假如现在要求在控制台中,输入被除数和除数,求商。

传统做法是这样的:

复制代码
1 public static void main(String[] args) {
2 System.out.println("请输入一个被除数:");
3 Scanner sc = new Scanner(System.in);
4 if(sc.hasNextInt()) {
5 int num1 = sc.nextInt();
6 System.out.println("请输入一个除数:");
7 if(sc.hasNextInt()) {
8 int num2 = sc.nextInt();
9 if(0 == num2) {
10 System.out.println("除数不能为0!");
11 }else {
12 int r = num1 / num2;
13 System.out.println("r = " + r);
14 }
15 }else {
16 System.out.println("除数输入不合法!");
17 }
18 }else {
19 //在控制台有可能输入字符串
20 System.out.println("被除数输入不合法!");
21 }
22
23 }
24
复制代码
从上面这个例子可以可以看出,这么简单的业务需求,代码也要写得那么长,因为要考虑的问题有很多,这样写代码会觉得很累,而且出现了异常,程序会中断,不会执行后面的代码。因此,Java编程语言使用异常处理机制为程序提供异常处理的能力。

1.2.2 java的异常处理
在Java中,异常处理的过程:

这种处理过程就像你上班遇到公路施工,你做出了处理——绕路行走,避开施工路段,让你按时到达公司!

2.异常的分类
在 Java 中,所有的异常都有一个共同的祖先 Throwable(可抛出)。Throwable 指定代码中可用异常传播机制通过 Java 应用程序传输的任何问题的共性。

Throwable: 有两个重要的子类:Exception(异常)和 Error(错误),二者都是 Java 异常处理的重要子类,各自都包含大量子类。

Error(错误):是程序无法处理的错误,表示运行应用程序中较严重问题。大多数错误与代码编写者执行的操作无关,而表示代码运行时 JVM(Java 虚拟机)出现的问题。例如,Java虚拟机运行错误(Virtual MachineError),当 JVM 不再有继续执行操作所需的内存资源时,将出现 OutOfMemoryError。这些异常发生时,Java虚拟机(JVM)一般会选择线程终止。

这些错误表示故障发生于虚拟机自身、或者发生在虚拟机试图执行应用时,如Java虚拟机运行错误(Virtual MachineError)、类定义错误(NoClassDefFoundError)等。这些错误是不可查的,因为它们在应用程序的控制和处理能力之 外,而且绝大多数是程序运行时不允许出现的状况。对于设计合理的应用程序来说,即使确实发生了错误,本质上也不应该试图去处理它所引起的异常状况。在 Java中,错误通过Error的子类描述。

Exception(异常):是程序本身可以处理的异常。Exception 类有一个重要的子类 RuntimeException。RuntimeException 类及其子类表示“JVM 常用操作”引发的错误。例如,若试图使用空值对象引用、除数为零或数组越界,则分别引发运行时异常(NullPointerException、ArithmeticException)和 ArrayIndexOutOfBoundException。注意:异常和错误的区别:异常能被程序本身可以处理,错误是无法处理。通常,Java的异常(包括Exception和Error)分为可查的异常(checked exceptions)和不可查的异常(unchecked exceptions)。

可查异常(编译器要求必须处置的异常):正确的程序在运行中,很容易出现的、情理可容的异常状况。可查异常虽然是异常状况,但在一定程度上它的发生是可以预计的,而且一旦发生这种异常状况,就必须采取某种方式进行处理。

除了RuntimeException及其子类以外,其他的Exception类及其子类都属于可查异常。这种异常的特点是Java编译器会检查它,也就是说,当程序中可能出现这类异常,要么用try-catch语句捕获它,要么用throws子句声明抛出它,否则编译不会通过。

不可查异常(编译器不要求强制处置的异常):包括运行时异常(RuntimeException与其子类)和错误(Error)。

Exception 这种异常分两大类运行时异常和非运行时异常(编译异常)。程序中应当尽可能去处理这些异常。

运行时异常:都是RuntimeException类及其子类异常,如NullPointerException(空指针异常)、IndexOutOfBoundsException(下标越界异常)等,这些异常是不检查异常,程序中可以选择捕获处理,也可以不处理。这些异常一般是由程序逻辑错误引起的,程序应该从逻辑角度尽可能避免这类异常的发生。运行时异常的特点是Java编译器不会检查它,也就是说,当程序中可能出现这类异常,即使没有用try-catch语句捕获它,也没有用throws子句声明抛出它,也会编译通过。

非运行时异常 (编译异常):是RuntimeException以外的异常,类型上都属于Exception类及其子类。从程序语法角度讲是必须进行处理的异常,如果不处理,程序就不能编译通过。如IOException、SQLException等以及用户自定义的Exception异常,一般情况下不自定义检查异常。

3.异常处理机制
通过上面对异常的解释,现在应该对异常有一定的了解。下面来说明在java中是如何处理异常的。

在java中用对象来表示异常的。
java是通过try-catch、try-catch-finally、try-catch-catch...语句来处理异常的。
3.1 try-catch
3.1.1 try-catch的使用
try{} 代码块用于执行可能存在异常的代码,catch(异常类型 异常对象的名称){}代码块用于捕获并处理异常。

复制代码
1 try{
2 //有可能出现异常的代码段1
3 //有可能出现异常的代码段2
4 }catch(异常类型 e){
5 //处理异常的代码段3
6 }
7 //代码段4
复制代码
处理1.2.1的两数相除问题的demo:

复制代码
1 public static void main(String[] args) {
2 Scanner sc = new Scanner(System.in);
3 try{
4 System.out.println("请输入一个被除数:");
5 int num1=sc.nextInt();
6 System.out.println("请输入一个除数:");
7 int num2=sc.nextInt();
8 int result=num1/num2;
9 }catch(Exception e){
10 System.out.println("在此处处理异常!");
11 }
12 System.out.println("程序运行结束");
13
14 }
复制代码
3.1.2 try-catch的执行顺序
第一种:没有遇到异常,即正常执行

第二种:匹配到异常

在try{}中的代码遇到异常时,会与catch()中括号里的异常进行比对,如果遇到的异常时属于catch的异常就会执行catch块中的代码,让后执行try-catch块后面的代码

第三种:匹配不到异常

3.2 try-catch-finally
try{} 代码块用于执行可能存在异常的代码,catch{}代码块用于捕获并处理异常。

finally{} 代码块用于回收资源(关闭文件、关闭数据库、关闭管道)的代码。finally代码块不管是否出现异常,都必须执行( finally块唯一不执行的情况System.exit(0) jvm正常退出。)

3.1.1 try-catch-finally的执行顺序
第一种:catch块没有return语句

(1)没有遇到异常:

try块内的代码——>finally块内的代码——>finally块后的代码

(2)遇到异常并匹配到异常:

try块内的代码——>catch块内的代码——>finally块内的代码——>finally块后的代码

复制代码
1 public class Test01 {
2 public static void main(String[] args) {
3 try {
4 int a=1/0;
5 System.out.println("try");
6 } catch (Exception e) {
7 System.out.println("catch");
8 }finally {
9 System.out.println("finally");
10 }
11 System.out.println("程序正常运行结束");
12 }
13 }
复制代码
结果:

1 catch
2 finally
3 程序正常运行结束

(3)遇到异常却没有匹配到异常:

try块内的代码——>finally块内的代码——>程序中断运行

复制代码
1 public static void main(String[] args) {
2 try {
3 int a=1/0;//会抛出ArithmeticException
4 System.out.println("try");
5 } catch (NullPointerException e) {
6 System.out.println("catch");
7 }finally {
8 System.out.println("finally");
9 }
10 System.out.println("程序正常运行结束");
11 }
复制代码
结果:

1 Exception in thread "main" finally
2 java.lang.ArithmeticException: / by zero
3 at Test1.Test01.main(Test01.java:8)
第二种:catch块有return语句

(1)没有异常

try块内的代码——>finally块内的代码——>try块内的return语句

复制代码
1 public class Test01 {
2 public static int test() {
3 try {
4 int a=2*2;
5 System.out.println("try");
6 return a;
7 } catch (Exception e) {
8 System.out.println("catch");
9 return 0;
10 }finally {
11 System.out.println("finally");
12 }
13 }
14 public static void main(String[] args) {
15 System.out.println(test());
16 System.out.println("程序正常运行结束");
17 }
18 }
复制代码
结果:

1 try
2 finally
3 4
4 程序正常运行结束
(2)遇到异常并匹配到异常:

执行顺序如图:

复制代码
1 public class Test01 {
2 public static int test() {
3 try {
4 int a=2/0;
5 System.out.println("try");
6 return a;
7 } catch (Exception e) {
8 System.out.println("catch");
9 return 0;
10 }finally {
11 System.out.println("finally");
12 }
13 }
14 public static void main(String[] args) {
15 System.out.println(test());
16 System.out.println("程序正常运行结束");
17 }
18 }
复制代码
结果:

1 catch
2 finally
3 0
4 程序正常运行结束
(3)遇到异常却没有匹配到异常:

try块内的代码——>finally块内的代码——>程序中断运行

复制代码
1 public class Test01 {
2 public static int test() {
3 try {
4 int a=2/0;
5 System.out.println("try");
6 return a;
7 } catch (NullPointerException e) {
8 System.out.println("catch");
9 return 0;
10 }finally {
11 System.out.println("finally");
12 }
13 }
14 public static void main(String[] args) {
15 System.out.println(test());
16 System.out.println("程序正常运行结束");
17 }
18 }
复制代码
结果:

finally
Exception in thread "main" java.lang.ArithmeticException: / by zero

at Test1.Test01.test(Test01.java:8)
at Test1.Test01.main(Test01.java:19)
  1. 声明异常
    4.1 throws

当开发者在定义方法时,事先知道方法在调用时会出现异常,但不知道该如何处理,此时可以在该方法上声明异常。表示该方法在调用过程中会出现异常,请调用者自行处理。

在java中使用throws 声明异常。一个方法可以声明多个异常,用,号分割,写法如下:

1 public void test2()throws IOException,RuntimeException{
2 //有异常出得代码,在此处没有处理
3 }
4.2 声明异常与方法重载的关系
声明异常和方法重载没有任何关系。

4.3 声明异常与方法重写的关系
如果父类方法声明了异常(检查时或运行时),子类方法可以完全遵循父类异常,也可以不声明异常。
如果父类方法没有声明异常,子类可以不声明异常,也可以声明RuntimeException,但不能声明Exception。
如果父类声明了运行时异常,子类可以完全遵循父类异常,也可以不声明异常。
5.抛出异常
当系统异常满足不了开发需要时,开发者可以自行根据需要自行抛出异常。

throw 用于手动抛出异常。

如果一直都没有处理(即没有用try-catch等语句)异常会把异常抛给调用者,一直抛到main函数处,如果在main函数中也没有处理继续在main函数后抛出异常,这时候会抛给jvm处理。如下栗子:

复制代码
1 public class Test01 {
2 public static void test1()throws IOException,RuntimeException{
3 //有异常抛出得代码,在此处没有处理,例如:throw new Exception(“异常信息”);
4 }
5 public static void test2()throws IOException,RuntimeException{
6 test1();//调用有抛出异常的方法,在此没有处理
7 }
8 public static void main(String[] args)throws IOException,RuntimeException {
9 test2();//main调用有抛出异常的方法,在此没有处理
10 }
11 }
复制代码
注意:

开发者根据自身需要可以选择抛出检查时异常和运行时异常

  1. 自定义异常
    当JDK 中的异常类型不能满足程序的需要时,可以自定义异常类。

自定义异常步骤:

[1] 确定异常类型.继承Excepion 或者RuntimeException
[2] 编写自定义异常类,并实现构造方法
[3] 在方法需要的地方手动声明并抛出异常。
复制代码
1 public class myException extends Exception {
2
3 public myException() {
4 super();
5 }
6
7 public myException(String message) {
8 super(message);
9 }
10
11 //自定义异常中的方法,以符合自己的需求
12 public void showInfo() {
13 System.out.println(super.getMessage()+"@Line:");
14 }
15 }
复制代码

原文地址https://www.cnblogs.com/ironHead-cjj/p/11337177.html

相关文章
|
2月前
|
Java
在 Java 中捕获和处理自定义异常的代码示例
本文提供了一个 Java 代码示例,展示了如何捕获和处理自定义异常。通过创建自定义异常类并使用 try-catch 语句,可以更灵活地处理程序中的错误情况。
76 1
|
2月前
|
Java API 调度
如何避免 Java 中的 TimeoutException 异常
在Java中,`TimeoutException`通常发生在执行操作超过预设时间时。要避免此异常,可以优化代码逻辑,减少不必要的等待;合理设置超时时间,确保其足够完成正常操作;使用异步处理或线程池管理任务,提高程序响应性。
88 12
|
2月前
|
监控 Java 应用服务中间件
高级java面试---spring.factories文件的解析源码API机制
【11月更文挑战第20天】Spring Boot是一个用于快速构建基于Spring框架的应用程序的开源框架。它通过自动配置、起步依赖和内嵌服务器等特性,极大地简化了Spring应用的开发和部署过程。本文将深入探讨Spring Boot的背景历史、业务场景、功能点以及底层原理,并通过Java代码手写模拟Spring Boot的启动过程,特别是spring.factories文件的解析源码API机制。
87 2
|
2月前
|
Java
在 Java 中,如何自定义`NumberFormatException`异常
在Java中,自定义`NumberFormatException`异常可以通过继承`IllegalArgumentException`类并重写其构造方法来实现。自定义异常类可以添加额外的错误信息或行为,以便更精确地处理特定的数字格式转换错误。
45 1
|
29天前
|
Java 程序员
深入理解Java异常处理机制
Java的异常处理是编程中的一块基石,它不仅保障了代码的健壮性,还提升了程序的可读性和可维护性。本文将深入浅出地探讨Java异常处理的核心概念、分类、处理策略以及最佳实践,旨在帮助读者建立正确的异常处理观念,提升编程效率和质量。
121 1
|
30天前
|
Java 开发者 UED
深入探索Java中的异常处理机制##
本文将带你深入了解Java语言中的异常处理机制,包括异常的分类、异常的捕获与处理、自定义异常的创建以及最佳实践。通过具体实例和代码演示,帮助你更好地理解和运用Java中的异常处理,提高程序的健壮性和可维护性。 ##
51 2
|
30天前
|
Java 开发者
Java中的异常处理机制深度剖析####
本文深入探讨了Java语言中异常处理的重要性、核心机制及其在实际编程中的应用策略,旨在帮助开发者更有效地编写健壮的代码。通过实例分析,揭示了try-catch-finally结构的最佳实践,以及如何利用自定义异常提升程序的可读性和维护性。此外,还简要介绍了Java 7引入的多异常捕获特性,为读者提供了一个全面而实用的异常处理指南。 ####
50 2
|
2月前
|
运维 Java 编译器
Java 异常处理:机制、策略与最佳实践
Java异常处理是确保程序稳定运行的关键。本文介绍Java异常处理的机制,包括异常类层次结构、try-catch-finally语句的使用,并探讨常见策略及最佳实践,帮助开发者有效管理错误和异常情况。
103 5
|
1月前
|
Java 程序员 UED
深入理解Java中的异常处理机制
本文旨在揭示Java异常处理的奥秘,从基础概念到高级应用,逐步引导读者掌握如何优雅地管理程序中的错误。我们将探讨异常类型、捕获流程,以及如何在代码中有效利用try-catch语句。通过实例分析,我们将展示异常处理在提升代码质量方面的关键作用。
45 3
|
1月前
|
Java 数据库连接 开发者
Java中的异常处理机制:深入解析与最佳实践####
本文旨在为Java开发者提供一份关于异常处理机制的全面指南,从基础概念到高级技巧,涵盖try-catch结构、自定义异常、异常链分析以及最佳实践策略。不同于传统的摘要概述,本文将以一个实际项目案例为线索,逐步揭示如何高效地管理运行时错误,提升代码的健壮性和可维护性。通过对比常见误区与优化方案,读者将获得编写更加健壮Java应用程序的实用知识。 --- ####