Java 异常处理以及如何捕获和处理多个异常

news2025/1/18 19:02:35

捕捉和处理异常

在Java中,我们使用异常处理程序组件try,catch和finally块来处理异常。

为了捕获和处理异常,我们将try...catch...finally代码块放置在可能产生异常的代码周围。finally块是可选的。

try...catch...finally的语法为:

try {
  // 代码
} catch (ExceptionType e) { 
  // 捕获块
} finally {
  //finally块
}

Java try ... catch块

可能会生成异常的代码放在try块中。

每个try块后面应紧跟着catch 或 finally块。发生异常时,它会被catch紧随其后的块捕获。

 catch块不能单独使用,必须紧随try块。

示例1:try ... catch块

class Main {
  public static void main(String[] args) {

    try {
      int divideByZero = 5 / 0;
      System.out.println("try块中的其余代码");
    } catch (ArithmeticException e) {
      System.out.println("ArithmeticException => " + e.getMessage());
    }

  }
}

输出结果

ArithmeticException => / by zero

在这个实例中

  • 我们在try块中将数字除以0。这产生一个ArithmeticException。

  • 发生异常时,程序将跳过try块中的其余代码。

  • 在这里,我们创建了一个catch块来处理ArithmeticException。因此,将catch执行块内的语句。

如果该try块中的所有语句均未生成异常,则跳过catch代码块。

多个捕获块

对于每个try块,可以有零个或多个catch块。

每个catch块的参数类型指示可以处理的异常类型。多个catch块使我们能够以不同方式处理每个异常。

示例2:多个捕获块

class ListOfNumbers {
  public int[] arrayOfNumbers = new int[10];

  public void writeList() {

    try {
      arrayOfNumbers[10] = 11;
    } catch (NumberFormatException e1) {
      System.out.println("NumberFormatException => " + e1.getMessage());
    } catch (IndexOutOfBoundsException e2) {
      System.out.println("IndexOutOfBoundsException => " + e2.getMessage());
    }

  }
}

class Main {
  public static void main(String[] args) {
    ListOfNumbers list = new ListOfNumbers();
    list.writeList();
  }
}

输出结果

IndexOutOfBoundsException => Index 10 out of bounds for length 10

在此示例中,我们声明了一个大小为10 的整数数组arrayOfNumbers。

我们知道数组索引总是从0开始。因此,当我们尝试为索引10分配一个值时,就会发生IndexOutOfBoundsException,因为数组arrayOfNumbers的边界是0到9。

当try块中发生异常时:

  • 异常被抛出给第一个catch块。第一个catch块不处理IndexOutOfBoundsException异常,因此它被传递给下一个catch块。

  • 上面示例中的第二个catch块是适当的异常处理程序,因为它处理IndexOutOfBoundsException。 因此,它被执行。

Java Finally块

对于每个try块,只能有一个finally块。

finally块是可选的。但是,如果已定义,它将始终执行(即使不会发生异常)。

如果发生异常,则在try...catch块之后执行。如果没有异常发生,则在try块之后执行。

finally块的基本语法为:

try {
  //code
} catch (ExceptionType1 e1) { 
  // catch 块
} catch (ExceptionType1 e2) {
 // catch 块
} finally {
  //finally块一直执行
}

示例3:finally块示例

class Main {
  public static void main(String[] args) {
    try {
      int divideByZero = 5 / 0;
    } catch (ArithmeticException e) {
      System.out.println("ArithmeticException => " + e.getMessage());
    } finally {
      System.out.println("Finally块总是执行");
    }
  }
}

输出结果

ArithmeticException => / by zero
Finally块总是执行

在此示例中,我们将数字除以0。这引发了一个ArithmeticException被catch块捕获,finally块始终执行。

使用finally块被认为是一种很好的做法。这是因为它包含了重要的清理代码,例如

  • 可能被return、continue或break语句意外跳过的代码

  • 关闭文件或连接

我们已经提到,finally总是执行,通常是这样的。但是,在某些情况下,finally块不执行:

  • 使用 System.exit()方法

  • finally块中发生异常

  • 线程被终止

示例4:try, catch 和finally示例

让我们举一个实例,我们尝试使用FileWriter创建一个新文件,并使用PrintWriter写入数据。

import java.io.*;

class ListOfNumbers {
  private int[] list = new int[10];

  public ListOfNumbers() {
    //在列表数组中存储整数值
    for (int i = 0; i < 10; i++) {
      list[i] = i;
    } 	
  }

}

  public void writeList() {
    PrintWriter out = null;

    try {
      System.out.println("进入try语句");

      //创建一个新文件OutputFile.txt
      out = new PrintWriter(new FileWriter("OutputFile.txt"));

      //将值从列表数组写入新创建的文件
      for (int i = 0; i < 10; i++) {
        out.println("Value at: " + i + " = " + list[i]);
      }
    } catch (IndexOutOfBoundsException e1) {
      System.out.println("IndexOutOfBoundsException => " + e1.getMessage());
    } catch (IOException e2) {
      System.out.println("IOException => " + e2.getMessage());
    } finally {
      //检查PrintWriter是否被打开
      if (out != null) {
        System.out.println("关闭PrintWriter");
        out.close();
      } else {
        System.out.println("PrintWriter无法打开");
      }
    }

  }
}

class Main {
  public static void main(String[] args) {
    ListOfNumbers list = new ListOfNumbers();
    list.writeList();
  }
}

当您运行此程序时,可能会发生两种可能性:

  1. try块中发生异常

  2. try块正常执行

创建新的FileWriter时可能会发生异常。 如果无法创建或写入指定的文件,则抛出IOException。

当发生异常时,我们将获得以下输出。

进入try语句
IOException => OutputFile.txt
PrintWriter无法打开

当未发生异常且该try块正常执行时,我们将获得以下输出。

进入try语句
关闭PrintWriter

将创建一个OutputFile.txt,并包含以下内容

Value at: 0 = 0
Value at: 1 = 1
Value at: 2 = 2
Value at: 3 = 3
Value at: 4 = 4
Value at: 5 = 5
Value at: 6 = 6
Value at: 7 = 7
Value at: 8 = 8
Value at: 9 = 9

try ... catch...finally 详细流程

让我们尝试在上述示例的帮助下详细了解异常处理的流程。

try ... catch...finally 详细流程

上图描述了在创建新FileWriter时发生异常时的程序执行流程。

  • 为了找到发生异常的方法,主方法调用writeList()方法,该方法随后调用FileWriter()方法来创建一个新的OutputFile.txt文件。

  • 发生异常时,运行时系统将跳过try块中的其余代码。

  • 它开始以相反的顺序搜索调用堆栈,以找到合适的异常处理程序。

  • 这里,FileWriter没有异常处理程序,因此运行时系统检查调用堆栈中的下一个方法,即writeList。

  • writeList方法有两个异常处理程序:一个处理IndexOutOfBoundsException,另一个处理IOException。

  • 然后,系统依次处理这些处理程序。

  • 此示例中的第一个处理程序处理IndexOutOfBoundsException。 这与try块引发的IOException不匹配。

  • 因此,检查下一个处理程序是哪个IOException处理程序。如与引发的异常类型匹配,因此将执行对应catch块中的代码。

  • 执行异常处理程序后,将执行finally块。

  • 在此场景中,由于FileWriter中发生了异常,所以PrintWriter对象out从未打开,因此不需要关闭。

现在,让我们假设在运行该程序时没有发生异常,并且try块正常执行。 在这种情况下,将创建并写入一个OutputFile.txt。

众所周知,finally块的执行与异常处理无关。由于没有异常发生,因此PrintWriter打开了并且需要关闭。这是通过finally块中的out.close()语句完成的。

捕获和处理多个异常

在Java 7之前,即使存在代码冗余,我们也必须针对不同类型的异常编写多个异常处理代码。

示例1:多个捕获块

class Main {
  public static void main(String[] args) {
    try {
      int array[] = new int[10];
      array[10] = 30 / 0;
    } catch (ArithmeticException e) {
      System.out.println(e.getMessage());
    } catch (ArrayIndexOutOfBoundsException e) {
      System.out.println(e.getMessage());
    } 
  }
}

输出结果

/ by zero

在此的示例可能会发生两个异常:

  • ArithmeticException -  因为我们试图将数字除以0。

  • ArrayIndexOutOfBoundsException -  因为我们已经声明了一个新的整数数组,数组边界为0到9,并且我们试图为索引10分配一个值。

我们在两个catch块中都打印出异常消息,即重复代码。

赋值运算符(=)的关联性是从右到左,因此首先将ArithmeticException与消息 / by zero起抛出。

在catch块中处理多个异常

现在可以在单个catch块中捕获多种类型的异常。

可以由catch块处理的每种异常类型都使用竖线(|)分隔。

其语法为:

try {
  // code
} catch (ExceptionType1 | Exceptiontype2 ex) { 
  // catch block
}

示例2:单个catch块中捕获多个异常

class Main {
  public static void main(String[] args) {
    try {
      int array[] = new int[10];
      array[10] = 30 / 0;
    } catch (ArithmeticException | ArrayIndexOutOfBoundsException e) {
      System.out.println(e.getMessage());
    }
  }
}

输出结果

/ by zero

在单个catch块中捕获多个异常,可以减少代码重复并提高效率。

编译该程序时生成的字节码将比具有多个catch块的程序小,因为没有代码冗余。

注意:如果一个catch块处理多个异常,则catch参数为隐式final。这意味着我们不能分配任何值来捕获参数。

捕获基本异常

当在单个catch块中捕获多个异常时,该规则将泛化为专门化规则。

这意味着,如果catch块中存在异常的层次结构,我们只能捕获基本异常,而不能捕获多个专门的异常。

让我们举个实例。

示例3:仅捕获基本异常类

class Main {
  public static void main(String[] args) {
    try {
      int array[] = new int[10];
      array[10] = 30 / 0;
    } catch (Exception e) {
      System.out.println(e.getMessage());
    }
  }
}

输出结果

/ by zero

我们知道所有异常类都是Exception类的子类。因此,我们不必捕获多个专门的异常,而只需捕获Exception类。

如果已经在catch块中指定了基本异常类,则不要在同catch一块中使用子异常类。否则,我们会得到一个编译错误。

让我们举个实例。

示例4:捕获基类和子异常类

class Main {
  public static void main(String[] args) {
    try {
      int array[] = new int[10];
      array[10] = 30 / 0;
    } catch (Exception | ArithmeticException | ArrayIndexOutOfBoundsException e) {
      System.out.println(e.getMessage());
    }
  }
}

输出结果

Main.java:6: error: Alternatives in a multi-catch statement cannot be related by subclassing

在此示例中,ArithmeticException和ArrayIndexOutOfBoundsException都是Exception类的子类。 因此,我们抛出一个编译错误。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/821703.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

重学C++系列之STL库

一、什么是STL库 STL是“Standard Template Library”的缩写&#xff0c;中文翻译为“标准模板库”。CSTL是一套功能强大的C模板类&#xff0c;提供了通用的模板类和函数&#xff0c;这些模板类和函数可以实现多种流行和常用的算法和数据结构&#xff0c;如字符串操作、链表、队…

mybatisJava对象、list和json转换

1. 参考mybatis-plus mybatis Java对象、list和json转换 网上好多不靠谱&#xff0c;参考mybatis-plus中TableField&#xff0c;mybatis中自定义实现 这样不需要对象中属性字符串接收&#xff0c;保存到表中&#xff0c;都是转义字符&#xff0c;使用时还要手动转换为对象或者…

【UI自动化测试】Jenkins配置

前一段时间帮助团队搭建了UI自动化环境&#xff0c;这里将Jenkins环境的一些配置分享给大家。 背景&#xff1a; 团队下半年的目标之一是实现自动化测试&#xff0c;这里要吐槽一下&#xff0c;之前开发的测试平台了&#xff0c;最初的目的是用来做接口自动化测试和性能测试&…

棱镜七彩成为“软件供应链安全推进工作组”首批成员单位

2023年7月29日&#xff0c;由基础软件质量控制与技术评价工业和信息化部重点实验室&#xff08;以下简称“实验室”&#xff09;举办的软件供应链安全研讨会暨软件供应链安全推进工作组成立会议在京成功召开。 会上&#xff0c;实验室委托中国软件评测中心&#xff08;工业和信…

助力质检维护,基于超轻量级分割模型ege-unet开发构建水泥基建裂缝分割识别系统

在前面的博文&#xff1a; 《参数量仅有50KB的超轻量级unet变种网络egeunet【参数和计算量降低494和160倍】医疗图像分割实践》 初步学习和实践了最新的超轻量级的unet变种网络在医疗图像领域内的表现&#xff0c;在上文中我们就说过会后续考虑将该网络模型应用于实际的生产业…

GNN+RA 文献阅读-- GNN对RA的建模

简述&#xff1a;主要是几篇 如何利用GNN 对 资源分配进行建模的paper&#xff0c;【1】【2】都是对无线链路建模&#xff0c;【3】比较有参考性&#xff0c;【4】偏于RL&#xff0c;对GNN表述模糊。 用GNN建模网络的思路&#xff1a; 1.Graph 是有向图还是无向图&#xff1f…

HarmonyOS 开发基础(二)组件拼凑简单登录页面

一、简单登录页面 Entry Component /* 组件可以基于struct实现&#xff0c;组件不能有继承关系&#xff0c;struct可以比class更加快速的创建和销毁。*/ struct Index {State message: string Hello Worldbuild() {// https://developer.harmonyos.com/cn/docs/documentation/…

万应低代码 7 月重点更新内容速递

速览版 详情版 低代码开发能力提升 业务逻辑 业务逻辑是什么&#xff1f; 在万应低代码中&#xff0c;「业务逻辑」指的是应用程序中的核心规则和功能&#xff0c;它决定了数据如何被处理和操作。就像搭积木一样&#xff0c;业务逻辑告诉计算机在特定情况下如何运行和响应。比…

【C语言】操作符----详解

&#x1f341; 博客主页:江池俊的博客 &#x1f4ab;收录专栏&#xff1a;C语言——探索高效编程的基石 &#x1f4bb; 其他专栏&#xff1a;数据结构探索 &#x1f4a1;代码仓库&#xff1a;江池俊的代码仓库 &#x1f3aa; 社区&#xff1a;C/C之家社区 &#x1f341; 如果觉…

【投资笔记】美股要变天了?

美股上涨&#xff0c;但是风险溢价创新低 7月&#xff0c;标普和纳指创两年依赖首次5个月连涨&#xff0c;但是风险溢价创二十年新低&#xff1b;https://asset.wsj.net/dynamic-insets/charts/cdc_5387e4742a1ca607d6defc38_embed.html 如果这个指标为正&#xff0c;说明投资…

视频媒体有哪些?视频媒体采访服务怎么做?

传媒如春雨&#xff0c;润物细无声&#xff0c;大家好&#xff0c;我是51媒体网胡老师。 一&#xff0c;在国内&#xff0c;主流的视频媒体包括&#xff1a; 1. 电视台&#xff1a;包括国家级、地方性和专业性电视频道&#xff0c;涵盖各类新闻、综艺、娱乐、体育等节目。 2…

链表OJ:环形链表

Lei宝啊&#xff1a;个人主页 愿所有美好与我们不期而遇 题目描述 &#xff1a; 给你一个链表的头节点 head &#xff0c;判断链表中是否有环。 接口&#xff1a; bool hasCycle(struct ListNode *head) 示例1&#xff1a; 示例2&#xff1a; 返回值&#xff1a; true或…

电力系统基础知识(东方电子)持续更新

文章目录 三相电、相电压、线电压断路器和继电器电压互感器、电流互感器GOOSE、SV 三相电、相电压、线电压 因为交流电可以通过变压器升降&#xff0c;很容易实现远距离输电&#xff0c;而直流电无法升降&#xff0c;远距离输电会造成巨大浪费。 三相电&#xff1a;三相交流电…

go程序使用tcp短连接报:only one usage of each socket address

环境及现象 Win10上位机&#xff08;C#,WPF&#xff09;后台使用go作为服务。 连接情况 C#连接大概60个TCP长连接&#xff08;设备&#xff09;。 后台go服务连接60个UDP短连接&#xff08;设备附属硬件&#xff09;&#xff0c; 10个TCP短连接&#xff08;PLC,modbus通讯&a…

【python】使用Selenium和Chrome WebDriver来获取 【腾讯云 Cloud Studio 实战训练营】中的文章信息

文章目录 前言导入依赖库设置ChromeDriver的路径创建Chrome WebDriver对象打开网页找到结果元素创建一个空列表用于存储数据遍历结果元素并提取数据提取标题、作者、发布时间等信息判断是否为目标文章提取目标文章的描述、阅读数量、点赞数量、评论数量等信息将提取的数据存储为…

坚鹏:中国邮储银行金融科技前沿技术发展与应用场景第2期培训

中国邮政储蓄银行金融科技前沿技术发展与应用场景第2期培训圆满结束 中国邮政储蓄银行拥有优良的资产质量和显著的成长潜力&#xff0c;是中国领先的大型零售银行。2016年9月在香港联交所挂牌上市&#xff0c;2019年12月在上交所挂牌上市。中国邮政储蓄银行拥有近4万个营业网点…

ByteBuffer

ByteBuffer 1.创建方式创建方式1:ByteBuffer buf ByteBuffer.allocate(int size);2.创建方式2:ByteBuffer buf ByteBuffer.allocateDirect(int size); 2.字符串转成ByteBuffer的3三种方式方式1: 采用put()方法,读数据时需要调用flip()切换为读模式方式2&#xff1a;以特定编码…

ChatGPT能否撰写科研论文?

ChatGPT&#xff0c;这款被许多人誉为语言处理领域的“黑马”&#xff0c;究竟能否应用于撰写科研论文&#xff1f;近期&#xff0c;以色列理工学院生物学家兼数据科学家Roy Kishony带领的团队&#xff0c;针对这一问题进行了系列研究&#xff0c;其结果已在《Nature》杂志上发…

MySQL表的内外连接

MySQL表的内外连接 一.内连接二.外连接1. 左外连接2. 右外连接 三.案例 表的连接分为内连和外连。 一.内连接 内连接实际上就是利用where子句对两种表形成的笛卡儿积进行筛选&#xff0c;我们前面学习的查询都是内连接&#xff0c;也是在开发过程中使用的最多的连接查询。而使…

使用 Go 语言实现二叉搜索树

原文链接&#xff1a; 使用 Go 语言实现二叉搜索树 二叉树是一种常见并且非常重要的数据结构&#xff0c;在很多项目中都能看到二叉树的身影。 它有很多变种&#xff0c;比如红黑树&#xff0c;常被用作 std::map 和 std::set 的底层实现&#xff1b;B 树和 B 树&#xff0c;…