SpringMVC 的三种异常处理方式详解

news2025/1/16 1:04:25

目录

1. 什么是异常

2. 为什么要全局异常处理

3. SpringMVC异常分类

4. 异常处理思路

5. 三种异常处理方式示例

① 配置 SimpleMappingExceptionResolver 处理器

② 实现 HandlerExceptionResolver 接口

③ 使用@ControllerAdvice+@ExceptionHandler实现全局异常

6. 响应封装类


1. 什么是异常

        见字如意,就是编码时所报的错误。异常主要分为两种:编译时异常和运行时异常RuntimeException。前者在编码时如不符合规范代码就会报红,是肉眼也看到的错,后者主要通过规范代码开发、测试通过手段减少运行时异常的发生。

        在Spring MVC中,异常处理是指对在请求处理过程中可能发生的异常情况进行捕获、处理和响应的机制。它能将所有类型的异常处理从各处理过程解耦出来,既保证了相关处理过程的功能较单一,也实现了异常信息的统一处理和维护。

2. 为什么要全局异常处理

        在开发中,不管是dao层、service层还是controller层,都有可能抛出异常。使用全局异常处理有以下几个优点:

  1. 统一的异常处理逻辑:通过全局异常处理,可以将应用程序中的异常处理逻辑集中到一个地方,避免在每个具体的业务代码中都编写相同或类似的异常处理代码。这样可以提高代码的可维护性,减少代码冗余,使代码更加清晰和易于理解。

  2. 提高系统的容错性:全局异常处理可以捕获应用程序中的未处理异常,避免异常的传递导致应用程序崩溃或出现未知错误。通过合理地处理异常,可以使系统在出现异常时保持稳定并继续正常运行,提高系统的容错性和健壮性。

  3. 统一的异常返回格式:全局异常处理可以定义统一的异常返回格式,将异常信息以一致的方式返回给客户端或用户。这样可以提供更友好和一致的错误提示,改善用户体验,并便于客户端对异常进行处理和展示。

  4. 安全性和安全漏洞的处理:全局异常处理可以捕获和处理应用程序中的安全漏洞或异常情况,例如未授权访问、非法操作等。通过适当的异常处理,可以防止一些潜在的安全问题,提高系统的安全性和防护能力。

  5. 异常日志记录和监控:全局异常处理可以统一记录应用程序中的异常信息,包括异常类型、发生位置、请求参数等,便于后续的异常分析、故障排查和系统监控。通过对异常进行有效的记录和监控,可以及时发现和解决潜在的问题,提高系统的可靠性和稳定性。

        总的来说,全局异常处理能够提高系统的可维护性、容错性和安全性,提供更好的用户体验,方便系统的监控和问题排查。在开发应用程序时,合理利用全局异常处理机制可以有效地处理异常情况,保证应用程序的稳定。

3. SpringMVC异常分类

在Spring MVC中,异常可以分为两类:应用程序异常和系统异常

  1. 应用程序异常:应用程序异常是指由应用程序自身业务逻辑引发的异常,通常是预期的异常情况。这些异常可以根据具体业务需求进行分类,例如用户输入验证失败、资源未找到、权限不足等。应用程序异常可以通过自定义异常类来表示,可以添加自定义的异常信息,以便更好地理解异常原因和处理异常。

  2. 系统异常:系统异常是指由系统运行环境或外部因素引发的异常,通常是非预期的异常情况。这些异常可能包括数据库连接失败、网络连接异常、服务器内部错误等。系统异常是无法通过应用程序自身的逻辑来解决的,需要通过合适的异常处理机制来捕获和处理。

        在Spring MVC中,可以通过合理的异常分类和处理机制来对应用程序异常和系统异常进行区分和处理。通常,应用程序异常可以通过@ExceptionHandler注解或自定义的异常处理器来处理,而系统异常可以通过全局异常处理器(HandlerExceptionResolver接口的实现)来捕获和处理。这样可以实现对不同类型异常的精细处理,提高系统的可靠性和用户体验。

4. 异常处理思路

        系统的dao、service、controller出现异常都通过throws Exception向上抛出,最后由springmvc前端控制器交由异常处理器进行异常处理。springmvc提供全局异常处理器(一个系统只有一个异常处理器)进行统一异常处理。

  1. 确定异常处理的范围和层级:首先,需要确定异常处理的范围,即异常处理应该放在哪个层级。在Spring MVC中,异常处理可以发生在Controller层、Service层或DAO层,具体取决于你的应用架构和需求。一般来说,应用程序异常可以在Controller层处理,而系统异常可以在全局异常处理器中处理。

  2. 定义异常类:根据你的应用程序需求,可以定义自定义的异常类来表示不同类型的异常。自定义异常类可以继承自ExceptionRuntimeException,并添加适当的属性和方法,以便更好地描述异常信息。例如,你可以定义一个ValidationException来表示用户输入验证失败的异常。

  3. 异常捕获和处理:在代码中,使用try-catch块来捕获可能发生的异常,并在catch块中进行相应的处理。对于应用程序异常,可以在Controller层的方法中使用@ExceptionHandler注解来捕获和处理特定类型的异常。对于系统异常,可以在全局异常处理器中实现HandlerExceptionResolver接口,并在resolveException方法中进行异常处理逻辑。

  4. 异常处理逻辑:在异常处理逻辑中,你可以根据具体的异常类型采取适当的处理措施。例如,对于应用程序异常,可以返回特定的错误页面或错误信息给用户,或者记录异常日志以便后续分析。对于系统异常,可以返回一个通用的错误页面或错误信息,或者进行一些系统级的异常处理操作。

  5. 异常信息传递和展示:在异常处理过程中,你可以将异常信息传递给前端页面或其他组件进行展示。可以通过ModelAndView对象、@ResponseBody注解或其他方式来传递异常信息。在展示异常信息时,可以根据具体需求进行格式化,以便用户能够清晰地理解异常原因。

  6. 单元测试和异常模拟:为了验证异常处理的正确性,建议编写相应的单元测试,并模拟各种异常情况。通过单元测试,可以确保异常处理逻辑的正确性和稳定性。

        总结来说,异常处理的思路是根据异常类型进行分类,确定异常处理的范围和层级,定义自定义异常类,捕获和处理异常,并根据具体情况进行异常处理逻辑和异常信息展示。通过合理的异常处理,可以提高系统的可靠性。

5. 三种异常处理方式示例

在演示下面之前我们现在没有进行异常处理事如何报错的:

1. 随便在那个方法或者controller手动写一个错误信息

2. 运行该路径

这就是没有经过任何异常处理的错误信息,如果将这个界面回显给客户当然是不可观的,请继续往下看。

① 配置 SimpleMappingExceptionResolver 处理器

        SpringMVC中自带了一个异常处理器叫SimpleMappingExceptionResolver,该处理器实现了HandlerExceptionResolver 接口,全局异常处理器都需要实现该接口。

1. spring-mvc.xml:

    <!-- springmvc提供的简单异常处理器 -->
    <bean class="org.springframework.web.servlet.handler.SimpleMappingExceptionResolver">
        <!-- 定义默认的异常处理页面 -->
        <property name="defaultErrorView" value="error"/>
        <!-- 定义异常处理页面用来获取异常信息的变量名,也可不定义,默认名为exception -->
        <property name="exceptionAttribute" value="xw"/>
        <!-- 定义需要特殊处理的异常,这是重要点 -->
        <property name="exceptionMappings">
            <props>
                <prop key="java.lang.RuntimeException">error</prop>
            </props>
            <!-- 还可以定义其他的自定义异常 -->
        </property>
    </bean>

注:页面跳转由SpringMVC来接管了,所以此处的定义默认的异常处理页面都应该配置成逻辑视图名。

2. 创建error.jsp

<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
    <title>Title</title>
</head>
<body>
<%--错误信息--%>
${xw}<br>
<img src="static/imgs/R.jpg">
</body>
</html>

配置异常处理后的界面:

② 实现 HandlerExceptionResolver 接口

1. 创建一个名为GlobalException的自定义异常类,它继承自RuntimeException类。

        在Java中,自定义异常类通常需要继承自Exception类或其子类。RuntimeException是Exception的一个子类,它表示在程序执行过程中可能会发生的异常情况。通过继承RuntimeException类,GlobalException可以被视为一个非受检异常(unchecked exception),即在方法签名中不需要显式声明或捕获。

        GlobalException类提供了多个构造函数,用于创建不同类型的异常对象。每个构造函数都调用了父类RuntimeException的相应构造函数,以便将异常信息传递给父类进行处理。

package com.ycxw.exception;

public class GlobalException extends RuntimeException {
    public GlobalException() {
    }

    public GlobalException(String message) {
        super(message);
    }

    public GlobalException(String message, Throwable cause) {
        super(message, cause);
    }

    public GlobalException(Throwable cause) {
        super(cause);
    }

    public GlobalException(String message, Throwable cause, boolean enableSuppression, boolean writableStackTrace) {
        super(message, cause, enableSuppression, writableStackTrace);
    }
}

        通过提供不同的构造函数,GlobalException类可以在不同的情况下创建异常对象,并传递相关的异常信息和原因。

        自定义异常类的好处在于可以根据应用程序的需求,定义特定类型的异常,以便更好地描述和处理不同的异常情况。在代码中,当某个方法或操作发生异常时,可以抛出GlobalException对象,并在上层进行相应的异常处理逻辑。

2. 定义了一个名为GlobalExceptionHandler的类,并实现了HandlerExceptionResolver接口。同时,使用@Component注解将该类标记为一个Spring组件,以便在应用程序中进行自动扫描和注册。

@Component
public class GlobalExceptionHandler implements HandlerExceptionResolver {
    @Override
    public ModelAndView resolveException(HttpServletRequest httpServletRequest,
                                         HttpServletResponse httpServletResponse,
                                         Object o, Exception e) {
        ModelAndView mv = new ModelAndView();
        mv.setViewName("error");
        if (e instanceof GlobalException){
            GlobalException globalException = (GlobalException) e;
            mv.addObject("xw",globalException.getMessage());
            mv.addObject("msg","全局异常....");
        }else if (e instanceof RuntimeException){
            RuntimeException runtimeException = (RuntimeException) e;
            mv.addObject("xw",runtimeException.getMessage());
            mv.addObject("msg","运行时异常....");
        }
        return mv;
    }
}
  1. 这段代码定义了一个名为GlobalExceptionHandler的类,并实现了HandlerExceptionResolver接口。同时,使用@Component注解将该类标记为一个Spring组件,以便在应用程序中进行自动扫描和注册。
  2. HandlerExceptionResolver接口是Spring MVC框架提供的用于处理异常的接口。通过实现该接口,可以自定义全局异常处理器,用于捕获和处理在请求处理过程中产生的异常。
  3. 在代码中,resolveException方法是HandlerExceptionResolver接口的一个方法,用于处理异常。它接受HttpServletRequest、HttpServletResponse、Object和Exception等参数,并返回一个ModelAndView对象。
  4. 在方法内部,首先创建了一个空的ModelAndView对象mv。然后,设置了视图的名称为"error",这意味着当发生异常时,将渲染名为"error"的视图。
  5. 接下来,通过instanceof判断异常的类型,进行不同的处理逻辑。如果异常是GlobalException类型,将异常信息和自定义的"全局异常...."消息添加到mv对象中。如果异常是RuntimeException类型,将异常信息和"运行时异常...."消息添加到mv对象中。
  6. 最后,将mv对象返回,以便框架将其渲染并返回给客户端。

        通过实现HandlerExceptionResolver接口并定义resolveException方法,可以自定义全局异常处理逻辑,并在应用程序中统一处理异常情况。这样可以提供更好的用户体验,并减少代码中重复的异常处理代码。

利用上一个报错信息:

3. 抛出一个全局错误信息

③ 使用@ControllerAdvice+@ExceptionHandler实现全局异常

        定义了一个名为GlobalExceptionResolver的类,并使用@ControllerAdvice注解将其标记为一个全局异常处理器。

package com.ycxw.exception;

import org.springframework.web.bind.annotation.ControllerAdvice;
import org.springframework.web.bind.annotation.ExceptionHandler;
import org.springframework.web.bind.annotation.ResponseBody;

import java.util.HashMap;
import java.util.Map;

@ControllerAdvice
public class GlobalExceptionResolver {

// 返回错误json数据
    @ResponseBody
    @ExceptionHandler
    public Map handler(Exception e){
        Map map = new HashMap();
        if (e instanceof GlobalException){
            GlobalException globalException = (GlobalException) e;
            map.put("xw",globalException.getMessage());
            map.put("msg","全局异常....");
        }else if (e instanceof RuntimeException){
            RuntimeException runtimeException = (RuntimeException) e;
            map.put("xw",runtimeException.getMessage());
            map.put("msg","运行时异常....");
        }else {
            map.put("xw",e.getMessage());
            map.put("msg","其它异常....");
        }
        return map;
    }
}
  1. @ControllerAdvice是Spring MVC框架提供的注解,用于定义全局控制器增强。通过使用@ControllerAdvice注解,可以将该类应用于所有的控制器中,并提供统一的异常处理逻辑。
  2. 在代码中,使用@ResponseBody注解将handler方法标记为返回JSON数据的方法。@ExceptionHandler注解用于指定该方法处理的异常类型。
  3. handler方法接受一个Exception类型的参数e,并返回一个Map对象。在方法内部,根据异常的类型进行不同的处理逻辑。
  4. 如果异常是GlobalException类型,将异常信息和自定义的"全局异常...."消息添加到map对象中。如果异常是RuntimeException类型,将异常信息和"运行时异常...."消息添加到map对象中。如果异常不属于上述两种类型,将异常信息和"其它异常...."消息添加到map对象中。
  5. 最后,将map对象作为JSON数据返回给客户端。

        通过使用@ControllerAdvice注解和@ExceptionHandler注解,可以自定义全局异常处理逻辑,并将异常信息以JSON格式返回给客户端。这样可以提供更好的错误处理和异常信息展示方式。

6. 响应封装类

        响应封装类的作用是将响应数据进行统一的封装和格式化,以便在应用程序中进行统一的处理和返回给客户端。

  • 通常,在一个应用程序中,不同的接口可能返回不同的数据结构和格式。这样会导致客户端在处理响应数据时需要编写大量的重复代码来解析和处理不同的响应格式,增加了开发和维护的复杂性。
  • 为了解决这个问题,可以使用响应封装类来对响应数据进行统一的封装。响应封装类通常是一个自定义的类,其中包含了与响应相关的信息,如状态码、消息、数据等。
  • 通过将响应数据封装到一个固定的数据结构中,可以使得所有的接口返回的响应数据格式保持一致。这样客户端只需要针对统一的响应数据结构进行处理,而不需要关注不同接口的具体返回格式。
  • 响应封装类还可以提供额外的功能,如异常处理、错误码定义、多语言支持等。通过在响应封装类中定义统一的异常处理逻辑,可以捕获和处理应用程序中的异常,并将异常信息封装到响应数据中返回给客户端。同时,可以在响应封装类中定义错误码,并根据错误码进行国际化处理,以支持多语言环境下的错误提示。

        总之,响应封装类的作用是提供一个统一的响应数据格式,简化客户端的处理逻辑,提供额外的功能如异常处理和多语言支持,并提高应用程序的可维护性和可扩展性。

例如: 

package com.ycxw.utils;

import java.util.HashMap;

public class R extends HashMap {
    public R data(String key, Object value) {
        this.put(key, value);
        return this;
    }
 
    public static R ok(int code, String msg) {
        R r = new R();
        r.data("success", true).data("code", code).data("msg", msg);
        return r;
    }
 
    public static R error(int code, String msg) {
        R r = new R();
        r.data("success", false).data("code", code).data("msg", msg);
        return r;
    }
 
    public static R ok(int code, String msg,Object data) {
        R r = new R();
        r.data("success", true).data("code", code).data("msg", msg).data("data", data);
        return r;
    }
 
    public static R ok(int code, String msg, long count, Object data) {
        R r = new R();
        r.data("success", true).data("code", code).data("msg", msg).data("count", count).data("data", data);
        return r;
    }
}

编写统一响应方法:

    // 响应封装类
    @ResponseBody
    @ExceptionHandler
    public R handler(Exception e){
        if (e instanceof GlobalException){
            GlobalException globalException = (GlobalException) e;
            return R.ok(505,"全局异常....",globalException.getMessage());
        }else if (e instanceof RuntimeException){
            RuntimeException runtimeException = (RuntimeException) e;
            return R.ok(505,"运行时异常....",runtimeException.getMessage());
        }else {
            return R.ok(505,"其他异常....",e.getMessage());
        }
    }

运行测试:

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

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

相关文章

TypeScript泛型和类型体操

&#x1f3ac; 岸边的风&#xff1a;个人主页 &#x1f525; 个人专栏 :《 VUE 》 《 javaScript 》 ⛺️ 生活的理想&#xff0c;就是为了理想的生活 ! 目录 泛型&#xff08;Generics&#xff09; 1. 泛型函数 2. 泛型接口 3. 泛型类 类型体操&#xff08;Type Gymnast…

leetcode 232 用栈实现队列

请你仅使用两个栈实现先入先出队列。队列应当支持一般队列支持的所有操作&#xff08;push、pop、peek、empty&#xff09;&#xff1a; 实现 MyQueue 类&#xff1a; void push(int x) 将元素 x 推到队列的末尾int pop() 从队列的开头移除并返回元素int peek() 返回队列开头…

Python实现天文计算

迷途小书童 读完需要 2分钟 速读仅需 1 分钟 1 简介 ephem 模块为 Python 提供了精确的天文计算能力&#xff0c;可以预测星球、卫星的轨道信息&#xff0c;计算日出日落、经星时间等数据&#xff0c;它的算法准确可靠。最初由 Brandon Craig Rhodes 在 20 世纪 90 年代开发&am…

数字IC验证23912--寄存器模型

文章目录 寄存器模型的集成总线UVC的实现总线UVC的示例Adapter的实现Adapter的集成 访问方式前门访问后门访问 寄存器模型的集成 总线UVC的实现 MCDF访问寄存器的总线接口时序较为简单。控制寄存器接口首先需要在每一个时钟解析cmd。当cmd为写指令时&#xff0c;即需要把数据c…

Linux安装mysql8.0.34(图文详细教程2023)

安装mysql数据库目录2023-09-13更新 1. 下载mysql数据库2. 安装3. mysql启动4. 进入数据库修改密码 以下是root用户操作&#xff0c; 非root用户&#xff0c;命令前请添加sudo 1. 下载mysql数据库 下载地址&#xff1a; https://dev.mysql.com/downloads/mysql/ 获取下载链接&…

OPENCV进行图像修复

API # -*- coding:utf-8 -*- """ 作者:794919561 日期:2023/9/14 """ import cv2 import numpy as npimg = cv2.imread("F:\\learnOpenCV\\openCVLearning\\pictures\\Lena.jpg") mask = cv2.imread

移动测试之语音识别功能如何测试?

移动测试之语音识别功能如何测试&#xff1f; 要知道语音识别功能如何测试&#xff0c;我们先了解智能产品语音交互流程&#xff1a; 所以&#xff0c;要进行测试的话&#xff0c;我们需要从以下几个维度来准备测试点&#xff1a; 基础功能测试&#xff1a; 1、声纹的录入&…

不定积分的概念和性质

目录 原函数 不定积分 不定积分的几何意义 原函数的存在定理 不定积分的性质 不定积分是微积分的一个关键部分&#xff0c;它涉及到一个函数的不定积分的计算。不定积分可以理解为求一个函数的原函数&#xff0c;也被称为反导数。原函数是一个函数&#xff0c;使得该函数的…

MongoDB-1入门介绍

NoSQL NoSQL(NoSQL Not Only SQL)&#xff0c;意即反SQL运动&#xff0c;指的是非关系型的数据库 优点 1、对数据库高并发读写。 2、对海量数据的高效率存储和访问。 3、对数据库的高可扩展性和高可用性。 弱点&#xff1a; 1、数据库事务一致性需求 2、数据库的写实时性…

多模态图像合成与编辑

由于信息在现实世界中以多种形式存在&#xff0c;多模态信息之间的有效交互和融合对于计算机视觉和深度学习研究中多模态数据的创建和感知起着关键作用。多模态图像合成与编辑由于具有强大的多模态信息交互建模能力&#xff0c;成为近年来的研究热点。多模态引导不是为网络训练…

点云从入门到精通技术详解100篇-从全局到局部的三维点云细节差异分析

目录 前言 国内外研究现状 细节差异分析相关研究 三维点云的相似性相关研究 存在的问题 三维点云对比的相关技术 2.1 三维点云的采集设备 2.2三维点云的存储格式 2.3三维点云的空间变换 2.4三维点云相似度分析 2.4.1点云特征的提取 2.4.2特征相似度计算 本文篇幅较长&#xff0…

解决java.util.NoSuchElementException

解决java.util.NoSuchElementException 解决java.util.NoSuchElementException摘要引言正文1. 了解异常的根本原因2. 避免不正确的索引3. 处理空集合4. 使用迭代器时要小心5. 异常处理 总结参考资料 博主 默语带您 Go to New World. ✍ 个人主页—— 默语 的博客&#x1f466;&…

博客-三更草堂

博客-三更草堂 后台分类页面修改&#xff0c;添加状态修改接口 ① …前端工程\sg-vue-admin\src\api\content\category.js 文件中添加新接口 // 修改分类状态 export function changeCategoryStatus(id, status) {const data {id,status}return request({url: /content/cat…

【编程实践】使用pcl提取给定点云的三维边界点

1 执行结果 原始点云可视化 搜索半径设置为0.1m 搜索半径设置为0.05m 2 代码实现 // boundary#include <pcl/point_types.h> #include <pcl/features/normal_3d.h> #include <pcl/features/boundary.h> #include <pcl/io/file_io.h> #include &l…

面对 HR 的空窗期提问,你会如何回答?

原文链接 面对 HR 的空窗期提问&#xff0c;你会如何回答&#xff1f; 你是否有过这样的经历&#xff0c;在一段时间内&#xff0c;你离开了工作岗位&#xff0c;或者在寻找新的工作机会&#xff0c;这段时间我们称之为“空窗期”。 对于这段时间&#xff0c;我们该如何看待&…

idea中的debug界面上没有进入方法的红色按钮

问题描述&#xff1a; 这里缺少进入系统方法的红色按钮。 问题解决方法&#xff1a; 在上面图片红框范围内右键点击进入。 点击号 搜索 ‘force’ 添加即可完成 上下拖动即可调整界面按钮顺序

I2C总线协议

什么是I2C I2C&#xff08;Inter-Integrated Circuit&#xff09;&#xff0c;也可以叫IIC、I2C&#xff0c;译作集成电路总线&#xff0c;是两线式串行通信总线&#xff0c;用于设备间的通讯等&#xff0c;标准情况下最高传送速率达100Kbps。顾名思义&#xff0c;I2C通讯只需…

如何用示波器测量放电波形

示波器那么重要&#xff0c;你确定不进来看看&#xff1f;_哔哩哔哩_bilibili 5分钟搞明白示波器的 带宽 采样率 存储深度_哔哩哔哩_bilibili 年轻人的第一台示波器选手持还是台式&#xff1f;_哔哩哔哩_bilibili 以示波器为例&#xff0c;测量某设备波形。 1、开机&#xff…

在Excel中使用SQL

说明: Excel中许多函数虽然能代替SQL的功能&#xff0c;但是比起SQL&#xff0c;还是有一些逊色&#xff0c;特意做了这个教程&#xff0c;主要有: 分组统计&#xff0c;Excel中用数据透视表&#xff0c;SQL中用Group By去重&#xff0c;Excel中可以用条件标识功能(开始->条…

Lyft 计划开源人工智能算法测试工具

共享租车公司 Lyft 投入大量时间和资源开发工具来测试机器学习算法&#xff0c;近日 Lyft 宣布将向公众开源这些工具。正式投入生产环境之前&#xff0c;一款机器学习算法的测试和更替非常耗费时间&#xff0c;Lyft 开发的测试工具&#xff0c;是属于一种模拟器&#xff0c;能够…