SpringBoot异常处理 ExceptionHandler的使用

2024-09-04 03:18

本文主要是介绍SpringBoot异常处理 ExceptionHandler的使用,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

应用程序在运行过程中,会有大量需要处理的异常。在页面解析的一个工程中,会存在多个service类同时出现页面解析异常和解析结果入库异常,而这就表示在程序中需要一个机制,去统一处理这些异常,提供统一的异常处理。因为我设计这个结构的主要目的是为了简化代码。

在探寻spring的异常处理机制的时候,主要有三种方式来统一处理异常。三种方式都是使用的@ExceptionHandler注解。

@ExceptionHandler注解解释:
当一个Controller中有方法加了@ExceptionHandler之后,这个Controller其他方法中没有捕获的异常就会以参数的形式传入加了@ExceptionHandler注解的那个方法中。

三种方式都需要首先为自己的系统设计一个自定义的异常类,通过它来传递状态码,以及一些其他参数信息。

public class ProcessException extends RuntimeException {private static final long serialVersionUID = 1L;// 任务明细IDprotected String rwmxid;public ProcessException(String rwmxid) {this.rwmxid = rwmxid;}public String getRwmxid() {return rwmxid;}public void setRwmxid(String rwmxid) {this.rwmxid = rwmxid;}}

第一种思路,设计一个基类。类中使用@ExceptionHandler注解 表明要做异常处理的方法

/*** Created by liuruijie.* 处理异常的类,需要处理异常的Controller直接继承这个类*/
public class BaseController {/*** 处理Controller抛出的异常* @param e 异常实例* @return Controller层的返回值*/@ExceptionHandler@ResponseBodypublic Object expHandler(Exception e){if(e instanceof SystemException){SystemException ex= (SystemException) e;return WebResult.buildResult().status(ex.getCode()).msg(ex.getMessage());}else{e.printStackTrace();return WebResult.buildResult().status(Config.FAIL).msg("系统错误");}}
}

这种方式的缺点为:之后所有需要异常处理的Controller都继承这个类,从而获取到异常处理的方法。
虽然这种方式可以解决问题,但是极其不灵活,因为动用了继承机制就只为获取一个默认的方法,这显然是不好的。

第二种方式,将这个基类变为接口,提供此方法的默认实现(也就是接口中的default方法,java8开始支持接口方法的默认实现)

/*** Created by liuruijie.* 接口形式的异常处理*/
public interface DataExceptionSolver {@ExceptionHandler@ResponseBodydefault Object exceptionHandler(Exception e){try {throw e;} catch (SystemException systemException) {systemException.printStackTrace();return WebResult.buildResult().status(systemException.getCode()).msg(systemException.getMessage());} catch (Exception e1){e1.printStackTrace();return WebResult.buildResult().status(Config.FAIL).msg("系统错误");}}
}

这种方式虽然没有占用继承,但是也不是很优雅,因为几乎所有的Controller都需要进行异常处理,于是我每个Controller都需要去写implement DataExceptionSolver,这显然不是我真正想要的。况且这种方式依赖java8才有的语法,这是一个很大的局限。

第三种方式,使用加强Controller做全局异常处理。

所谓加强Controller就是@ControllerAdvice注解,有这个注解的类中的方法的某些注解会应用到所有的Controller里,其中就包括@ExceptionHandler注解。
于是可以写一个全局的异常处理类:

/**
* @ClassName ExceptionHandle 
* @Description 统一处理控制层的异常
* @date 2020年3月14日 上午9:55:41 
* @version 1.0*/
@ControllerAdvice
public class ExceptionHandle {private final Logger log = LoggerFactory.getLogger(getClass());@ExceptionHandler(ParseException.class)public void parseException(ParseException ex) {ex.printStackTrace();log.error("parseException ",ExceptionUtil.getMessage(ex));Map<String,Object> paraMap = new HashMap<String,Object>();paraMap.put("rwmxid", ex.getRwmxid());CommonUtil.updateDataLog(paraMap);}@ExceptionHandler(ProcessException.class)public void processException(ProcessException ex) {ex.printStackTrace();log.error("ProcessException ",ExceptionUtil.getMessage(ex));Map<String,Object> paraMap = new HashMap<String,Object>();paraMap.put("rwmxid", ex.getRwmxid());CommonUtil.updateDataLog(paraMap);}@ExceptionHandler(Exception.class)@ResponseBodypublic TransEntity<?> error(Exception ex) {ex.printStackTrace();log.error("Exception ",ExceptionUtil.getMessage(ex));return TransEntity.error();}}

如此,我们现在的Controller中的方法就可以很简洁了:

/*** Created by liuruijie on 2016/12/28.* 账号*/
@RestController
@RequestMapping("passport")
public class PassportController {PassportService passportService;@RequestMapping("login")public Object doLogin(HttpSession session, String username, String password){User user = passportService.doLogin(username, password);session.setAttribute("user", user);return WebResult.buildResult().redirectUrl("/student/index");}
}

在passprotService的doLogin方法中,可能会抛出用户名或密码错误等异常,然后就会交由ExceptionHandle 去处理,直接返回异常信息给前端,然后前端也不需要关心是否返回了异常,因为这些都已经定义好了。 

如果我们自定义了异常,也可以在指定抛出我们自定义的异常,然后在全局异常处理类中进行处理, @ExceptionHandler(ParseException.class) 表明,会处理抛出的 ParseException 异常。

 

 

 

这篇关于SpringBoot异常处理 ExceptionHandler的使用的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

Spring事务传播机制最佳实践

《Spring事务传播机制最佳实践》Spring的事务传播机制为我们提供了优雅的解决方案,本文将带您深入理解这一机制,掌握不同场景下的最佳实践,感兴趣的朋友一起看看吧... 目录1. 什么是事务传播行为2. Spring支持的七种事务传播行为2.1 REQUIRED(默认)2.2 SUPPORTS2

怎样通过分析GC日志来定位Java进程的内存问题

《怎样通过分析GC日志来定位Java进程的内存问题》:本文主要介绍怎样通过分析GC日志来定位Java进程的内存问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、GC 日志基础配置1. 启用详细 GC 日志2. 不同收集器的日志格式二、关键指标与分析维度1.

Java进程异常故障定位及排查过程

《Java进程异常故障定位及排查过程》:本文主要介绍Java进程异常故障定位及排查过程,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、故障发现与初步判断1. 监控系统告警2. 日志初步分析二、核心排查工具与步骤1. 进程状态检查2. CPU 飙升问题3. 内存

Linux中压缩、网络传输与系统监控工具的使用完整指南

《Linux中压缩、网络传输与系统监控工具的使用完整指南》在Linux系统管理中,压缩与传输工具是数据备份和远程协作的桥梁,而系统监控工具则是保障服务器稳定运行的眼睛,下面小编就来和大家详细介绍一下它... 目录引言一、压缩与解压:数据存储与传输的优化核心1. zip/unzip:通用压缩格式的便捷操作2.

java中新生代和老生代的关系说明

《java中新生代和老生代的关系说明》:本文主要介绍java中新生代和老生代的关系说明,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、内存区域划分新生代老年代二、对象生命周期与晋升流程三、新生代与老年代的协作机制1. 跨代引用处理2. 动态年龄判定3. 空间分

Java设计模式---迭代器模式(Iterator)解读

《Java设计模式---迭代器模式(Iterator)解读》:本文主要介绍Java设计模式---迭代器模式(Iterator),具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,... 目录1、迭代器(Iterator)1.1、结构1.2、常用方法1.3、本质1、解耦集合与遍历逻辑2、统一

Java内存分配与JVM参数详解(推荐)

《Java内存分配与JVM参数详解(推荐)》本文详解JVM内存结构与参数调整,涵盖堆分代、元空间、GC选择及优化策略,帮助开发者提升性能、避免内存泄漏,本文给大家介绍Java内存分配与JVM参数详解,... 目录引言JVM内存结构JVM参数概述堆内存分配年轻代与老年代调整堆内存大小调整年轻代与老年代比例元空

深度解析Java DTO(最新推荐)

《深度解析JavaDTO(最新推荐)》DTO(DataTransferObject)是一种用于在不同层(如Controller层、Service层)之间传输数据的对象设计模式,其核心目的是封装数据,... 目录一、什么是DTO?DTO的核心特点:二、为什么需要DTO?(对比Entity)三、实际应用场景解析

Java 线程安全与 volatile与单例模式问题及解决方案

《Java线程安全与volatile与单例模式问题及解决方案》文章主要讲解线程安全问题的五个成因(调度随机、变量修改、非原子操作、内存可见性、指令重排序)及解决方案,强调使用volatile关键字... 目录什么是线程安全线程安全问题的产生与解决方案线程的调度是随机的多个线程对同一个变量进行修改线程的修改操

从原理到实战深入理解Java 断言assert

《从原理到实战深入理解Java断言assert》本文深入解析Java断言机制,涵盖语法、工作原理、启用方式及与异常的区别,推荐用于开发阶段的条件检查与状态验证,并强调生产环境应使用参数验证工具类替代... 目录深入理解 Java 断言(assert):从原理到实战引言:为什么需要断言?一、断言基础1.1 语