【年薪百万之IT界大神成长之路】JAVA 中的异常处理,阅后即焚!!!

2024-03-15 22:30

本文主要是介绍【年薪百万之IT界大神成长之路】JAVA 中的异常处理,阅后即焚!!!,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

 
愿你如阳光,明媚不忧伤。

目録

    • 1. 异常定义
    • 2. 异常类型
    • 3. 异常方法
    • 4. 异常关键字
    • 5. 内置异常
    • 6. 简单的自定义异常
    • 7. 高级的自定义异常
  • 【每日一面】
          • try-catch-finally 中,如果 catch 中 return 了,finally 还会执行吗?在finally中return好吗?

 


1. 异常定义

Exception 异常指的是在程序运行过程中发生的异常事件(编译时产生的不是异常,而是错误),通常是由外部问题(如硬件错误、输入错误)所导致的。它阻止了程序按照程序员的预期正常执行。在Java等面向对象的编程语言中异常属于对象

在这里插入图片描述

 


2. 异常类型

Throwable类包括了Error类和Exception类,异常类有两个主要的子类:IOException 类和 RuntimeException 类。

三种类型的异常:

  1. 检查性异常:最具代表的检查性异常是用户错误或问题引起的异常,这是程序员无法预见的。例如要打开一个不存在文件时,一个异常就发生了,这些异常在编译时不能被简单地忽略。
  2. 运行时异常: 运行时异常是可能被程序员避免的异常。与检查性异常相反,运行时异常可以在编译时被忽略
  3. 错误: 错误不是异常,而是脱离程序员控制的问题。错误在代码中通常被忽略。例如,当栈溢出时,一个错误就发生了,它们在编译也检查不到的。

可以看到 Java 语言定义了很多的异常类在 java.lang 标准包中。实在是太多了,整理不过来,具体遇到时再具体分析。

在这里插入图片描述

 


3. 异常方法

下面的列表是 Throwable 类的主要( public )方法:

方法说明
Throwable()构建一个空的 Throwable ,随后可被 initCause 初始化
Throwable(String message)构建一个带有详细信息的 Throwable ,随后可被 initCause 初始化
Throwable(String message, Throwable cause)构建一个带有详细信息和导致原因的 Throwable
Throwable(Throwable cause)构建一个带有导致原因的 Throwable
final synchronized void addSuppressed(Throwable exception)记录被抑制的异常(finally语句块抛出异常,try语句块中的异常会丢失)
synchronized Throwable fillInStackTrace()清空原来的栈内的trace信息,然后在当前的调用位置处重新建立trace信息
synchronized Throwable getCause()返回一个Throwable 对象异常的导致原因
String getLocalizedMessage()返回关于发生异常的本地化详细信息
String getMessage()返回关于发生异常的详细信息
StackTraceElement[] getStackTrace()返回一个包含堆栈层次的数组。下标为0的元素代表栈顶,最后一个元素代表方法调用堆栈的栈底
final synchronized Throwable[] getSuppressed()获取被抑制的异常
synchronized Throwable initCause(Throwable cause)初始化导致原因,最多可调用一次
void printStackTrace()打印toString()结果和栈层次到System.err,即错误输出流
void printStackTrace(PrintStream s)打印toString()结果和栈层次到标准错误流
void printStackTrace(PrintWriter s)打印toString()结果和栈层次到指定文件
void setStackTrace(StackTraceElement[] stackTrace)设置将返回的堆栈跟踪元素,高级系统使用
String toString()返回 Throwable 的字符串表示形式

 


4. 异常关键字

finally语句块总是会被执行。只有finally块,执行完成之后,才会回来执行try或者catch块中的return或者throw语句,如果finally中使用了return或者throw等终止方法的语句,则不会跳回执行,直接停止。
主方法上也可以使用throws抛出。如果在主方法上使用了throws抛出,就表示在主方法里面可以不用强制性进行异常处理,如果出现了异常,就交给JVM进行默认处理,则此时会导致程序中断执行。

关键字说明
try用于监听异常
catch用于捕获异常
finally用于回收在try块里打开的资源(数据库连接,网络连接,磁盘文件等)
throw用于抛出异常
throws用在方法签名中,用于声明该方法可能抛出的异常

 


5. 内置异常

  • java.lang.ArithmeticException
    算数异常,例如除数不能为0
package com.ITGodRoad.exception;public class ExceptionDemo {public static void main(String[] args) {int a = 5;int b = 0;System.out.println(a / b);}
}-----------------------------------------------------------------
・【运行结果】
Exception in thread "main" java.lang.ArithmeticException: / by zeroat com.ITGodRoad.exception.ExceptionDemo.main(ExceptionDemo.java:7)

Eclipse 中异常快捷键:Alt+Shift+Z

现在我们用try-catch监听并捕获异常

public class ExceptionDemo {public static void main(String[] args) {int a = 5;int b = 0;try {System.out.println(a / b);} catch (ArithmeticException e) {System.out.println("除数不能为0");e.printStackTrace();}}
}-----------------------------------------------------------------
・【运行结果】
除数不能为0
java.lang.ArithmeticException: / by zeroat com.ITGodRoad.exception.ExceptionDemo.main(ExceptionDemo.java:8)

coverage 代码覆盖率 → 无论是否发生异常,finally 代码块中的代码总会被执行。
绿色:代码被执行过
黄色:代码部分被执行过
红色:代码没有被执行过

在这里插入图片描述
throw 主动抛出异常,可以看到虽然捕获的是 ArithmeticException 算数异常,但是控制台信息打印的是我们主动抛出的 RuntimeException 运行时异常。

public class ExceptionDemo {public static void main(String[] args) {int a = 5;int b = 0;try {System.out.println(a / b);} catch (ArithmeticException e) {throw new RuntimeException();} finally {System.out.println("这是一个有趣的世界!");}}
}-----------------------------------------------------------------
・【运行结果】
这是一个有趣的世界!
Exception in thread "main" java.lang.RuntimeExceptionat com.ITGodRoad.exception.ExceptionDemo.main(ExceptionDemo.java:10)

throws 向上抛出异常,用在函数上,声明该函数的功能可能会出现问题,将异常抛出,使问题暴露出来,用于处理。可以抛给虚拟机处理,或者使用 try…catch… 进行处理。虚拟机的处理方式,就是将异常打印出来,并且将在异常处的代码终止

public class ExceptionDemo {public static void main(String[] args) {int a = 5;int b = 0;System.out.println(devision(a, b));}public static int devision(int a, int b) throws ArithmeticException {return a / b;}
}-----------------------------------------------------------------
・【运行结果】
Exception in thread "main" java.lang.ArithmeticException: / by zeroat com.ITGodRoad.exception.ExceptionDemo.devision(ExceptionDemo.java:11)at com.ITGodRoad.exception.ExceptionDemo.main(ExceptionDemo.java:7)

 


6. 简单的自定义异常

在 Java 中你可以自定义异常。如果要自定义异常类,扩展Exception类即可,这样的自定义异常都属于检查异常(checked exception),编译时要求进行 try-catch 处理。如果要自定义非检查异常,则扩展RuntimeException。

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

  1. 一个无参构造函数
  2. 一个带有String参数的构造函数,并传递给父类的构造函数。
  3. 一个带有String参数和Throwable参数,并都传递给父类构造函数
  4. 一个带有Throwable 参数的构造函数,并传递给父类的构造函数。

Alt+shift+S 选择从父类生成构造器即可自动生成上面的构造函数

package com.ITGodRoad.exception;public class ExceptionDemo {public static void main(String[] args) {int a = 5;int b = 0;int result = 0;try {result = devision(a, b);} catch (MyCompileException e) {e.printStackTrace();}System.out.println("程序还可以继续运行,result:" + result);System.out.println("****************************************");updatePassword("123456");}
-----------------------------------------------------------------public static int devision(int a, int b) throws MyCompileException {if (b == 0) {throw new MyCompileException("除数不能为0");}return a / b;}
-----------------------------------------------------------------public static String updatePassword(String password) {if (password.equals("123456")) {throw new MyRuntimeException("密码不能过于简单");}return password;}
}
-----------------------------------------------------------------
class MyCompileException extends Exception {/****/private static final long serialVersionUID = 1L;public MyCompileException() {super();// TODO Auto-generated constructor stub}public MyCompileException(String message, Throwable cause, boolean enableSuppression, boolean writableStackTrace) {super(message, cause, enableSuppression, writableStackTrace);// TODO Auto-generated constructor stub}public MyCompileException(String message, Throwable cause) {super(message, cause);// TODO Auto-generated constructor stub}public MyCompileException(String message) {super(message);// TODO Auto-generated constructor stub}public MyCompileException(Throwable cause) {super(cause);// TODO Auto-generated constructor stub}}
-----------------------------------------------------------------
class MyRuntimeException extends RuntimeException {/****/private static final long serialVersionUID = 1L;public MyRuntimeException() {super();// TODO Auto-generated constructor stub}public MyRuntimeException(String message, Throwable cause, boolean enableSuppression, boolean writableStackTrace) {super(message, cause, enableSuppression, writableStackTrace);// TODO Auto-generated constructor stub}public MyRuntimeException(String message, Throwable cause) {super(message, cause);// TODO Auto-generated constructor stub}public MyRuntimeException(String message) {super(message);// TODO Auto-generated constructor stub}public MyRuntimeException(Throwable cause) {super(cause);// TODO Auto-generated constructor stub}}-----------------------------------------------------------------
・【运行结果】
com.ITGodRoad.exception.MyCompileException: 除数不能为0at com.ITGodRoad.exception.ExceptionDemo.devision(ExceptionDemo.java:23)at com.ITGodRoad.exception.ExceptionDemo.main(ExceptionDemo.java:10)
程序还可以继续运行,result:0
****************************************
Exception in thread "main" com.ITGodRoad.exception.MyRuntimeException: 密码不能过于简单at com.ITGodRoad.exception.ExceptionDemo.updatePassword(ExceptionDemo.java:30)at com.ITGodRoad.exception.ExceptionDemo.main(ExceptionDemo.java:18)

可以看出,自定义异常,会将异常的处理权交给自己,而不是 JVM ,这样就避免了 JVM 遇到一些级别低的错误而使整个程序关闭。

在这里插入图片描述
在这里插入图片描述

 


7. 高级的自定义异常

可以自定义属性,例如添加异常级别(警告 warn 、错误 error 等),异常类型(业务异常、操作异常等),等其他自定义信息,还可以指定异常记录的输出位置,这样以后查看 log 就可以很轻松的定位程序问题所在了。

  • NormalException
package com.ITGodRoad.exception;import java.util.Map;import com.ITGodRoad.constant.Constant;public class NormalException extends RuntimeException {/****/private static final long serialVersionUID = 1L;private String level;private String errorType;private Map<String, String> data;private boolean isMessage;public Map<String, String> getData() {return data;}public void setData(Map<String, String> data) {this.data = data;}public String getLevel() {return level;}public String getErrorType() {return errorType;}public void setLevelError() {this.level = Constant.LEVEL_ERROR;}public void setErrorTypeBusiness() {this.errorType = Constant.ERROR_TYPE_BUSINESS;}public boolean isMessage() {return isMessage;}public NormalException(String errorMessageCode) {super(errorMessageCode);this.level = Constant.LEVEL_WARN;this.errorType = Constant.ERROR_TYPE_COMMUNICATION;this.data = null;this.isMessage = false;}// 判断是否是直接输出信息,false 则去数据库查询public NormalException(String errorMessageCode, boolean isMessage) {super(errorMessageCode);this.level = Constant.LEVEL_WARN;this.errorType = Constant.ERROR_TYPE_COMMUNICATION;this.data = null;this.isMessage = isMessage;}
}
  • MessageMasterService

@Service
@Transactional
public class MessageMasterService extends BaseService {@Autowiredprivate MessageMasterDao messageMasterDao;public String getMessage(String codeId) {log.info("codeId:" + codeId);if (Util.isEmptyString(codeId)) {return MessageCodes.DEFAULT_ERROR_DETAIL;}try {MessageMaster messageMaster = messageMasterDao.getMessageMasterById(Constant.IT_GOD_ROAD_WEB_SITE, codeId);if (Util.isNull(messageMaster)) {messageMaster = messageMasterDao.getMessageMasterById(Constant.IT_GOD_ROAD_WEB_SITE, MessageCodes.DEFAULT_ERROR);}if (Util.isNull(messageMaster)) {return "";}return messageMaster.getMessageDesc();} catch (Exception e) {return MessageCodes.DEFAULT_ERROR_DETAIL;}}
}
  • MessageMasterDao
@Repository
public class MessageMasterDao extends BaseDao {public MessageMaster getMessageMasterById(String type, String id) {log.info("type:" + type);log.info("id:" + id);MessageMasterId messageMasterId = new MessageMasterId();messageMasterId.setMessageType(type);messageMasterId.setMessageId(id);Session session = sessionFactory.getCurrentSession();MessageMaster messageMaster = session.get(MessageMaster.class, messageMasterId);return messageMaster;}}
  • MessageMaster

/*** MessageMaster generated by hbm2java*/
@Entity
@Table(name = "MESSAGE_MASTER")
public class MessageMaster implements java.io.Serializable {/*** */private static final long serialVersionUID = 1L;private MessageMasterId id;private String messageDesc;public MessageMaster() {}public MessageMaster(MessageMasterId id, String messageDesc) {this.id = id;this.messageDesc = messageDesc;}@EmbeddedId@AttributeOverrides({@AttributeOverride(name = "messageType", column = @Column(name = "MESSAGE_TYPE", nullable = false, length = 32)),@AttributeOverride(name = "messageId", column = @Column(name = "MESSAGE_ID", nullable = false, length = 32)) })public MessageMasterId getId() {return this.id;}public void setId(MessageMasterId id) {this.id = id;}@Column(name = "MESSAGE_DESC", nullable = false, length = 1024)public String getMessageDesc() {return this.messageDesc;}public void setMessageDesc(String messageDesc) {this.messageDesc = messageDesc;}}

 


【每日一面】

try-catch-finally 中,如果 catch 中 return 了,finally 还会执行吗?在finally中return好吗?

注意!!! 在 finally 中改变返回值的做法是不好的,因为如果存在 finally 代码块,try中的 return 语句不会立马返回调用者,而是记录下返回值待 finally 代码块执行完毕之后再向调用者返回其值,然后如果在 finally 中修改了返回值,就会返回修改后的值。显然,在 finally 中返回或者修改返回值会对程序造成很大的困扰,C#中直接用编译错误的方式来阻止程序员干这种龌龊的事情,Java 中也可以通过提升编译器的语法检查级别来产生警告或错误。
catch 中 return 了 finally 还是会执行,在 return 前执行

在这里插入图片描述

这篇关于【年薪百万之IT界大神成长之路】JAVA 中的异常处理,阅后即焚!!!的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/813454

相关文章

详解Java如何向http/https接口发出请求

《详解Java如何向http/https接口发出请求》这篇文章主要为大家详细介绍了Java如何实现向http/https接口发出请求,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 用Java发送web请求所用到的包都在java.net下,在具体使用时可以用如下代码,你可以把它封装成一

SpringBoot使用Apache Tika检测敏感信息

《SpringBoot使用ApacheTika检测敏感信息》ApacheTika是一个功能强大的内容分析工具,它能够从多种文件格式中提取文本、元数据以及其他结构化信息,下面我们来看看如何使用Ap... 目录Tika 主要特性1. 多格式支持2. 自动文件类型检测3. 文本和元数据提取4. 支持 OCR(光学

Java内存泄漏问题的排查、优化与最佳实践

《Java内存泄漏问题的排查、优化与最佳实践》在Java开发中,内存泄漏是一个常见且令人头疼的问题,内存泄漏指的是程序在运行过程中,已经不再使用的对象没有被及时释放,从而导致内存占用不断增加,最终... 目录引言1. 什么是内存泄漏?常见的内存泄漏情况2. 如何排查 Java 中的内存泄漏?2.1 使用 J

JAVA系统中Spring Boot应用程序的配置文件application.yml使用详解

《JAVA系统中SpringBoot应用程序的配置文件application.yml使用详解》:本文主要介绍JAVA系统中SpringBoot应用程序的配置文件application.yml的... 目录文件路径文件内容解释1. Server 配置2. Spring 配置3. Logging 配置4. Ma

Java 字符数组转字符串的常用方法

《Java字符数组转字符串的常用方法》文章总结了在Java中将字符数组转换为字符串的几种常用方法,包括使用String构造函数、String.valueOf()方法、StringBuilder以及A... 目录1. 使用String构造函数1.1 基本转换方法1.2 注意事项2. 使用String.valu

java脚本使用不同版本jdk的说明介绍

《java脚本使用不同版本jdk的说明介绍》本文介绍了在Java中执行JavaScript脚本的几种方式,包括使用ScriptEngine、Nashorn和GraalVM,ScriptEngine适用... 目录Java脚本使用不同版本jdk的说明1.使用ScriptEngine执行javascript2.

Spring MVC如何设置响应

《SpringMVC如何设置响应》本文介绍了如何在Spring框架中设置响应,并通过不同的注解返回静态页面、HTML片段和JSON数据,此外,还讲解了如何设置响应的状态码和Header... 目录1. 返回静态页面1.1 Spring 默认扫描路径1.2 @RestController2. 返回 html2

Spring常见错误之Web嵌套对象校验失效解决办法

《Spring常见错误之Web嵌套对象校验失效解决办法》:本文主要介绍Spring常见错误之Web嵌套对象校验失效解决的相关资料,通过在Phone对象上添加@Valid注解,问题得以解决,需要的朋... 目录问题复现案例解析问题修正总结  问题复现当开发一个学籍管理系统时,我们会提供了一个 API 接口去

Java操作ElasticSearch的实例详解

《Java操作ElasticSearch的实例详解》Elasticsearch是一个分布式的搜索和分析引擎,广泛用于全文搜索、日志分析等场景,本文将介绍如何在Java应用中使用Elastics... 目录简介环境准备1. 安装 Elasticsearch2. 添加依赖连接 Elasticsearch1. 创

Spring核心思想之浅谈IoC容器与依赖倒置(DI)

《Spring核心思想之浅谈IoC容器与依赖倒置(DI)》文章介绍了Spring的IoC和DI机制,以及MyBatis的动态代理,通过注解和反射,Spring能够自动管理对象的创建和依赖注入,而MyB... 目录一、控制反转 IoC二、依赖倒置 DI1. 详细概念2. Spring 中 DI 的实现原理三、