Java异常处理

简介: Java异常处理

@TOC


# 前言 编写程序时,经常要在可能出现的错误的地方加上检测代码,过多的分支会导致程序的代码加长臃肿,可读性差,所以才采用异常的处理机制。 Java的异常处理机制,把可能出现异常的代码集中起来,和正常代码分开,使得程序简洁、优雅,便于维护 。
# 一、异常的认识 ## 1.异常的体系结构 ![在这里插入图片描述](https://ucc.alicdn.com/images/user-upload-01/5a757dc037e144a1bc4b516b8f4f6d56.png?x-oss-process=image/watermark,type_d3F5LXplbmhlaQ,shadow_50,text_Q1NETiBAcXFfNTI3OTcxNzA=,size_20,color_FFFFFF,t_70,g_se,x_16#pic_center) ![在这里插入图片描述](https://ucc.alicdn.com/images/user-upload-01/bba37ba6b35b452ea212013e5f1f2098.png?x-oss-process=image/watermark,type_d3F5LXplbmhlaQ,shadow_50,text_Q1NETiBAcXFfNTI3OTcxNzA=,size_11,color_FFFFFF,t_70,g_se,x_16#pic_center) 红色:编译时异常 蓝色:运行时异常 ## 2.常见的异常 > 1. FileNotFoundException(文件没有找到异常) > 通常来说,当我们要读取文件的时候,如果找不到对应的文件,就会报异常。 > > 2. ClassNotFoundException(指定的类不存在) > 比如:当试图将一个**String**类型数据转换为指定的**数字类型**,但该字符串**不满足**数值型数据的要求时,就抛出这个异常。 > > 例如将**String**类型的数据"123456"转换为**数值型**数据时,可以转换。 > 但是如果**String**类型的数据中包含了**非数字型**的字符,比如"123456", > 此时转换为数值型时就会出现异常。系统就会捕捉到这个异常,并进行处理 > > 2. NullPointerException(空指针异常) > 这类异常经常出现,比如调用了未经初始化的对象或者是不存在的对象。当我们在创建对象或数数的时候可能会出现。 > 4. ArrayIndexOutOfBoundsException(数组角标越界异常) > 我们在创建数组的时候,要记得数组是从**0**号位置开始的,最后一位的下标是数组长度-1. > > 3. ClassCastException(数据类型转换异常) > 当试图将对某个对象强制执行向下转换,但该对象又不可转换或又不可转换为其子类的实例时将出现该异常 **下面是一个数据类型转换异常的例子:** ```java Object obj = new Date(); String str = (String)obj; ``` > 6. NumberFormatException(数字格式化异常) 例子: ```java String str = "123"; str = "abc"; int num = Integer.parseInt(str); ``` > 7. ArithmeticException(算数运算异常) 比如:除零运算 # 二、异常的处理:抓抛模型 ## 2.1 “抛” 程序在正常执行的过程中,一旦出现异常,就会在异常代码处生成一个对应异常类的对象,并将此对象抛出。一旦抛出对象以后,其后的代码就不再执行。 关于异常对象的产生:① 系统自动生成的异常对象 ② 手动的生成一个异常对象,并抛出(throw) 代码如下(示例): ## 2.2"抓" > 可以理解为异常的处理方式:① try-catch-finally ② throws # 三、异常的处理方式 ## 3.1 方式一: try-catch-finally ### 3.1.1使用方式 ```java @Test public void test1(){ String str = "123"; str = "abc"; int num = 0; try{ num = Integer.parseInt(str); System.out.println("hello-----1"); }catch(NumberFormatException e){ // System.out.println("出现数值转换异常了,不要着急...."); //String getMessage(): // System.out.println(e.getMessage()); //printStackTrace(): e.printStackTrace(); }catch(NullPointerException e){ System.out.println("出现空指针异常了,不要着急...."); }catch(Exception e){ System.out.println("出现异常了,不要着急...."); } System.out.println(num); System.out.println("hello-----2"); } } ``` ### 3.1.2 注意点 > 1.finally是可选的,不一定要写 > > 2.finally中声明的代码是一定会被执行的代码,即使catch中又出现异常,try中有return语句,catch中有return语句等情况 > > 3.像数据库连接、输入输出流、网络编程Socket等资源,JVM是不能自动的回收的,我们需要自己手动的进行资源的释放。此时的资源释放,就需要声明在finally中。 ## 3.2 方式二:throws 1. 如果父类中被重写的方法没有throws方式处理异常,则子类重写的方法也不能使用throws 。 2. 如果子类重写的方法中有异常,必须使用**try-catch-finally**方式处理。 3. 执行的方法a中,先后又调用了另外的几个方法,这几个方法是递进 关系执行的。我们建议这几个方法使用**throws**的方式进行处理。 而执行的方法a可以考虑使用try-catch-finally方式进行处理。 ## 3.3 两种异常处理方式的选择 > 1. 如果父类中被重写的方法没有throws方式处理异常,则子类重写的方法也不能使用throws,意味着如果 子类重写的方法中有异常,必须使用try-catch-finally方式处理。 > 2. 执行的方法a中,先后又调用了另外的几个方法,这几个方法是递进关系执行的。我们建议这几个方法使用 throws的方式进行处理。而执行的方法a可以考虑使用try-catch-finally方式进行处理。 ## 3.4 如何自定义异常类 1. 继承于现有的异常结构:RuntimeException 、Exception 2. 提供全局常量:serialVersionUID 3. 提供重载的构造器 ```java public class MyException extends Exception{ static final long serialVersionUID = -7034897193246939L; public MyException(){ } public MyException(String msg){ super(msg); } } ``` ## 3.5 throw和throws关键字的区别 1、写法上 : throw 在方法体内使用,throws 函数名后或者参数列表后方法体前. 2、意义 : throw 强调动作,而throws 表示一种倾向、可能但不一定实际发生 3.throws: 后面跟的是异常类,可以一个,可以多个,多个用逗号隔开。 throw 后跟的是异常对象,或者异常对象的引用。 4.throws: 用户抛出异常,当在当前方法中抛出异常后,当前方法执行结束 (throws 后,如果有finally语句的话,会执行到finally语句后再结束。)。 可以理解成return一样。 1、定义 : 一个方法不处理这个异常,而是调用层次向上传递,谁调用这个方法,这个异常就由谁来处理。 2、throw : 将产生的异常抛出(强调的是动作),抛出的既可以是异常的引用 也可以是异常对象。(位置: 方法体内) 3、throws : 如果一个方法可能会出现异常,但没有能力处理这种异常, 可以在方法声明处用throws子句来声明抛出异常。用它修饰的方法向调用者 表明该方法可能会抛出异常(可以是一种类型,也可以是多种类型, 用逗号隔开)(位置: 写在方法名 或方法名列表之后,在方法体之前。) **注意** : > 调用可能会抛出异常的方法,必须添加try-catch代码块尝试去捕获异常 或者 添加throws 声明 来将异常 > 抛出给更上一层的调用者进行处理,这里需要注意一个细节:新的异常包含原始异常的所有信息,根据这个我们可以去追溯最初异常发生的位置, # 总结 本文对异常做了简要的总结,希望可以帮到大家,有什么不完善或者错误的地方,大家可以给出意见。 #
相关文章
|
3月前
|
安全 Java
Java异常处理:程序世界的“交通规则
Java异常处理:程序世界的“交通规则
347 98
|
3月前
|
安全 Java 编译器
驾驭Java异常处理:从新手到专家的优雅之道
驾驭Java异常处理:从新手到专家的优雅之道
248 59
|
6月前
|
Java 编译器 数据库连接
Java异常处理:写出更健壮的代码
Java异常处理:写出更健壮的代码
222 0
|
5月前
|
Java 数据库 C++
Java异常处理机制:try-catch、throws与自定义异常
本文深入解析Java异常处理机制,涵盖异常分类、try-catch-finally使用、throw与throws区别、自定义异常及最佳实践,助你写出更健壮、清晰的代码,提升Java编程能力。
|
7月前
|
Java 程序员 数据库连接
我们详细地讲解一下 Java 异常及要如何处理
我是小假 期待与你的下一次相遇 ~
171 1
|
Java 程序员
Java编程中的异常处理:从基础到高级
在Java的世界中,异常处理是代码健壮性的守护神。本文将带你从异常的基本概念出发,逐步深入到高级用法,探索如何优雅地处理程序中的错误和异常情况。通过实际案例,我们将一起学习如何编写更可靠、更易于维护的Java代码。准备好了吗?让我们一起踏上这段旅程,解锁Java异常处理的秘密!
|
8月前
|
Java
java 多线程异常处理
本文介绍了Java中ThreadGroup的异常处理机制,重点讲解UncaughtExceptionHandler的使用。通过示例代码展示了当线程的run()方法抛出未捕获异常时,JVM如何依次查找并调用线程的异常处理器、线程组的uncaughtException方法或默认异常处理器。文章还提供了具体代码和输出结果,帮助理解不同处理器的优先级与执行逻辑。
204 1
|
10月前
|
SQL Java 中间件
【YashanDB知识库】yasdb jdbc驱动集成BeetISQL中间件,业务(java)报autoAssignKey failure异常
在BeetISQL 2.13.8版本中,客户使用batch insert向yashandb表插入数据并尝试获取自动生成的sequence id时,出现类型转换异常。原因是beetlsql在prepareStatement时未指定返回列,导致yashan JDBC驱动返回rowid(字符串),与Java Bean中的数字类型tid不匹配。此问题影响业务流程,使无法正确获取sequence id。解决方法包括:1) 在batchInsert时不返回自动生成的sequence id;2) 升级至BeetISQL 3,其已修正该问题。
【YashanDB知识库】yasdb jdbc驱动集成BeetISQL中间件,业务(java)报autoAssignKey failure异常
|
10月前
|
SQL druid Oracle
【YashanDB知识库】yasdb jdbc驱动集成druid连接池,业务(java)日志中有token IDENTIFIER start异常
客户Java日志中出现异常,影响Druid的merge SQL功能(将SQL字面量替换为绑定变量以统计性能),但不影响正常业务流程。原因是Druid在merge SQL时传入null作为dbType,导致无法解析递归查询中的`start`关键字。
|
10月前
|
运维 Java 程序员
Java中的异常处理方法
本文深入剖析Java异常处理机制,介绍可检查异常、运行时异常和错误的区别与处理方式。通过最佳实践方法,如使用合适的异常类型、声明精确异常、try-with-resources语句块、记录异常信息等,帮助开发者提高代码的可靠性、可读性和可维护性。良好的异常处理能保证程序稳定运行,避免资源泄漏和潜在问题。
305 5