iis服务器助手广告广告
返回顶部
首页 > 资讯 > 精选 >springBoot2.X配置全局捕获异常的方法
  • 627
分享到

springBoot2.X配置全局捕获异常的方法

2023-06-20 17:06:37 627人浏览 泡泡鱼
摘要

本篇内容主要讲解“SpringBoot2.X配置全局捕获异常的方法”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“springBoot2.X配置全局捕获异常的方法”吧!springBoot2.X配

本篇内容主要讲解“SpringBoot2.X配置全局捕获异常的方法”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习springBoot2.X配置全局捕获异常的方法”吧!

springBoot2.X配置全局捕获异常

先来看一段代码:当传入的id是0的时候,就会报异常。

@RestControllerpublic class HelloController {    @GetMapping("/getUser")    public String getUser(int id) {        int j = 1 / id;        return "SUCCESS" + j;    }}

访问时:

springBoot2.X配置全局捕获异常的方法

我们知道这个页面要是给用户看到,用户可能不知道这是什么。

方法一:将异常捕获

@GetMapping("/getUser")    public String getUser(int id) {        int j;        try {            j = 1 / id;        } catch (Exception e) {            return "系统异常";        }        return "SUCCESS" + j;    }

这种方法当然可以,但是当我们有很多方法时,需要在每个方法上都加上。

哎,太鸡肋了吧。

那么都没有全局的拦截处理呢?

当然了

方法二:通过@ControllerAdvice注解配置

@ControllerAdvice(basePackages = "com.yiyang.myfirstspringdemo.controller")public class GlobalExceptionHandler {    @ExceptionHandler(RuntimeException.class)    @ResponseBody    public Map<String,Object> errorResult()  {        Map<String, Object> map = new HashMap<>();        map.put("errorCode", "500");        map.put("errORMsg", "全局捕获异常");        return map;    }}
  • @ExceptionHandler 表示拦截异常

  • @ControllerAdvice 是 controller 的一个辅助类,最常用的就是作为全局异常处理的切面类

  • @ControllerAdvice 可以指定扫描范围

注意:下面还需要在启动类上加上,否则诶呦效果

package com.yiyang.myfirstspringdemo;import org.springframework.boot.SpringApplication;import org.springframework.boot.autoconfigure.SpringBootApplication;@SpringBootApplication(scanBasePackages = {"com.yiyang.myfirstspringdemo.error", "com.yiyang.myfirstspringdemo.controller"})public class MyFirstSpringDemoApplication {    public static void main(String[] args) {        SpringApplication.run(MyFirstSpringDemoApplication.class, args);    }}

在启动类上,将扫描包范围controller和全局异常处理类,加上去。

springBoot2.X配置全局捕获异常的方法

这样当我们在访问的时候,出现的异常提示信息就是我们在全局异常处理中设置的返回值。

springboot2.x 全局异常处理的正确方式

WEB项目中,异常堆栈信息是非常敏感的。因此,需要一个全局的异常处理,捕获异常,给客户端以友好的错误信息提示。基于 Spring Boot 很容易实现全局异常处理。

相关jar依赖引入

<!-- Spring Boot 启动父依赖 --><parent>    <groupId>org.springframework.boot</groupId>    <artifactId>spring-boot-starter-parent</artifactId>    <version>2.1.6.RELEASE</version>    <relativePath/> <!-- lookup parent from repository --></parent><properties>    <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>    <project.reporting.outputEncoding>UTF-8</project.reporting.outputEncoding>    <java.version>1.8</java.version></properties><dependencies>    <!-- Spring Boot Web 依赖 -->    <dependency>        <groupId>org.springframework.boot</groupId>        <artifactId>spring-boot-starter-web</artifactId>    </dependency></dependencies>

全局异常控制器

package com.yb.demo.common.handler;import com.yb.demo.common.enums.CodeEnum;import com.yb.demo.common.exception.BizException;import com.yb.demo.pojo.response.Result;import lombok.extern.slf4j.Slf4j;import org.apache.commons.lang3.StringUtils;import org.springframework.validation.BindException;import org.springframework.web.HttpRequestMethodNotSupportedException;import org.springframework.web.bind.MethodArgumentNotValidException;import org.springframework.web.bind.MissingServletRequestParameterException;import org.springframework.web.bind.annotation.ExceptionHandler;import org.springframework.web.bind.annotation.RestControllerAdvice;import javax.validation.ValidationException;import java.util.StringJoiner;@Slf4j@RestControllerAdvicepublic class GlobalExceptionHandler {        @ExceptionHandler(BizException.class)    public Result handleBizException(BizException ex) {        Result result = Result.renderErr(ex.getCode());        if (StringUtils.isNotBlank(ex.getRemark())) {            result.withRemark(ex.getRemark());        }        return result;    }        @ExceptionHandler(BindException.class)    public Result handleBindException(BindException ex) {        StringJoiner sj = new StringJoiner(";");        ex.getBindingResult().getFieldErrors().forEach(x -> sj.add(x.getDefaultMessage()));        return Result.renderErr(CodeEnum.REQUEST_ERR).withRemark(sj.toString());    }        @ExceptionHandler(ValidationException.class)    public Result handleValidationException(ValidationException ex) {        return Result.renderErr(CodeEnum.REQUEST_ERR).withRemark(ex.getCause().getMessage());    }        @ExceptionHandler(MethodArgumentNotValidException.class)    public Result handleMethodArgumentNotValidException(MethodArgumentNotValidException ex) {        StringJoiner sj = new StringJoiner(";");        ex.getBindingResult().getFieldErrors().forEach(x -> sj.add(x.getDefaultMessage()));        return Result.renderErr(CodeEnum.REQUEST_ERR).withRemark(sj.toString());    }        @ExceptionHandler(MissingServletRequestParameterException.class)    public Result handleMissingServletRequestParameterException(MissingServletRequestParameterException ex) {        return Result.renderErr(CodeEnum.REQUEST_ERR).withRemark(ex.getMessage());    }        @ExceptionHandler(value = HttpRequestMethodNotSupportedException.class)    public Result handleHttpRequestMethodNotSupportedException(HttpRequestMethodNotSupportedException ex) {        return Result.renderErr(CodeEnum.METHOD_NOT_ALLOWED);    }        @ExceptionHandler(value = Exception.class)    public Result handleException(Exception ex) {        log.error(ex.getMessage(), ex);        return Result.renderErr(CodeEnum.SERVER_ERR);    }}

个性化异常处理

自定义异常

在实际web开发过程中,往往会遇到在某些场景下需要终止当前流程,直接返回。那么,通过抛出自定义异常,并在全局异常中捕获,用以友好的提示客户端。

@Datapublic class BizException extends RuntimeException {    private static final long serialVersionUID = 1L;    private CodeEnum code;    private String remark;    public BizException(CodeEnum code) {        super(code.getMessage());        this.code = code;    }    public BizException withRemark(String remark) {        this.remark = remark;        return this;    }}

使用方式如下:

    public Student1DO addStudent(Student1DO student) {        if (StringUtils.isNotBlank(student.getStudName())) {            // 举例扔个业务异常,实际使用过程中,应该避免扔异常            throw new BizException(CodeEnum.REQUEST_ERR).withRemark("studName不能为空");        }        student1Mapper.insert(student);        return student;    }

返回效果如下:

{
"code": -400,
"msg": "请求错误(studName不能为空)",
"data": {},
"ttl": 0
}

根据阿里巴巴规范,流程控制还是不要通过抛异常的方式。在正常开发过程中,应避免使用这种方式。

【强制】异常不要用来做流程控制,条件控制,因为异常的处理效率比条件分支低。

使用 Spring validation 完成数据后端校验

定义实体类,加上validation相关注解

package com.yb.demo.pojo.model.db1;import com.baomidou.mybatisplus.annotation.FieldFill;import com.baomidou.mybatisplus.annotation.TableField;import com.baomidou.mybatisplus.annotation.TableName;import lombok.Data;import javax.validation.constraints.Min;import javax.validation.constraints.Size;@Data@TableName("student")public class Student1DO {    private Long id;    @Size(max = 8, message = "studName长度不能超过8")    private String studName;    @Min(value = 12, message = "年龄不能低于12岁")    private Integer studAge;    private String studSex;    @TableField(fill = FieldFill.INSERT)    private Integer createTime;    @TableField(fill = FieldFill.INSERT_UPDATE)    private Integer updateTime;}

在Controller 方法上加上 @Validated 注解

@PostMapping("/student/add")public Result addStudent(@Validated @RequestBody Student1DO student) {    student = studentService.addStudent(student);    return Result.renderOk(student);}

实际效果如下:

{
"code": -400,
"msg": "请求错误(年龄不能低于12岁)",
"data": {},
"ttl": 0
}

结束语

具体代码见:https://GitHub.com/daoshenzzg/springboot2.x-example

到此,相信大家对“springBoot2.X配置全局捕获异常的方法”有了更深的了解,不妨来实际操作一番吧!这里是编程网网站,更多相关内容可以进入相关频道进行查询,关注我们,继续学习!

--结束END--

本文标题: springBoot2.X配置全局捕获异常的方法

本文链接: https://www.lsjlt.com/news/298869.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

本篇文章演示代码以及资料文档资料下载

下载Word文档到电脑,方便收藏和打印~

下载Word文档
猜你喜欢
  • c++中int和double有什么区别
    int 和 double 是 c++ 的数据类型,用于表示整数和浮点数。它们的关键区别在于:1. 范围:int 为整数,double 为浮点数且范围更大;2. 存储大小:int 占 4 ...
    99+
    2024-05-14
    c++ 隐式转换
  • C++ 多线程程序测试的挑战和策略
    多线程程序测试面临不可重复性、并发错误、死锁和缺乏可视性等挑战。策略包括:单元测试:针对每个线程编写单元测试,验证线程行为。多线程模拟:使用模拟框架在控制线程调度的情况下测试程序。数据竞...
    99+
    2024-05-14
    c++ 多线程
  • c++中深拷贝和浅拷贝的应用时间
    浅拷贝复制对象指针或引用,仅适用于不含动态分配内存或简单数据结构的对象;深拷贝复制实际数据,包括动态分配内存,适用于包含动态分配内存或复杂数据结构的对象。 浅拷贝和深拷贝的应用时间 在...
    99+
    2024-05-14
    c++
  • 探索用于 C++ 服务器架构的高级数据结构
    在 c++++ 服务器架构中,选择适当的高级数据结构至关重要。哈希表用于快速数据查找,树用于表示数据层次结构,图用于表示对象之间的关系。这些数据结构在实践中有着广泛的应用,例如缓存系统、...
    99+
    2024-05-14
    c++ 数据结构 社交网络 键值对
  • fixed在c++中的作用
    fixed 关键字在 c++ 中用于将浮点数存储为固定小数,提供更高精度,尤其适用于需要高精度的金融计算。fixed 将浮点数表示为具有固定小数位数的小数,默认情况下使用十进制表示法,小...
    99+
    2024-05-14
    c++
  • insert在c++中怎么用
    insert() 函数在 c++ 中用于在容器(如 vector、set)中插入元素,提供了一种动态调整容器大小并添加新元素的方法。它需要两个参数:要插入元素的位置 (pos) 和要插入...
    99+
    2024-05-14
    c++ 标准库
  • 如何使用 Golang 构建 RESTful API 并处理 JSON 响应?
    如何使用 golang 构建和处理 json 响应的 restful api步骤:创建 golang 项目并安装 gorilla mux。定义路由并处理 http 请求。安装 json ...
    99+
    2024-05-14
    golang git
  • c++中int和long的区别
    int 和 long 都是 c++ 中的整型类型,主要区别在于范围和存储空间:范围:int 为 32 位整数,范围为 [-2^31, 2^31-1];long 为 64 位整数,范围为 ...
    99+
    2024-05-14
    c++ 数据丢失
  • c++中int a(n)和int a[n]的区别
    int a(n)声明一个不可变的整型变量,而int a[n]声明一个可修改元素的整型数组,用于存储和处理数据序列或集合。 int a(n) 和 int a[n] 在 C++ 中的区别 ...
    99+
    2024-05-14
    c++
  • C++ 多线程编程中调试和故障排除的技术
    c++++ 多线程编程的调试技巧包括:使用数据竞争分析器检测读写冲突,并使用同步机制(如互斥锁)解决。使用线程调试工具检测死锁,并通过避免嵌套锁和使用死锁检测机制来解决。使用数据竞争分析...
    99+
    2024-05-14
    c++ 多线程 故障排除 同步机制
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作