SpringBoot + thymeleaf 捕捉校验参数异常并统一处理

Spring Boot + thymeleaf 捕捉校验参数异常并统一处理

捕捉校验失败异常信息类

import com.fyyc.jhyzm.blog.mayday.exception.BlogException;
import com.fyyc.jhyzm.blog.mayday.reuse.Data;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.context.support.DefaultMessageSourceResolvable;
import org.springframework.validation.BindException;
import org.springframework.web.bind.MethodArgumentNotValidException;
import org.springframework.web.bind.annotation.ControllerAdvice;
import org.springframework.web.bind.annotation.ExceptionHandler;
import org.springframework.web.servlet.ModelAndView;
import javax.servlet.http.HttpServletRequest;
import javax.validation.ConstraintViolation;
import javax.validation.ConstraintViolationException;
import java.util.stream.Collectors;

/**
 * @ ClassName: ControllerExceptionHandler
 * @ Author: longxin
 * @ CreatTime: 2020/4/13 0013 17:35
 * @ version: 1.0
 */
//加了改注解会拦截所有抛出的异常
@ControllerAdvice
public class ControllerExceptionHandler {
	//异常接收的前端页面
    private final static String error_path = "error/error";
    //日志
    private final Logger logger = LoggerFactory.getLogger(this.getClass());
    
    //BlogException 自定义通用异常处理
    @ExceptionHandler(BlogException.class)
    public ModelAndView exceptionHander(HttpServletRequest request, BlogException e) throws Exception {
        logger.error("路由 : {},Blog错误信息 : {}", request.getRequestURL(), e.getStatus() + " " + e.getMessage());
        //Data - 我定义的通用数据模型 
        Data<String> data = new Data(e.getMessage(), e.getCode(), false, request.getRequestURL());

        ModelAndView mv = new ModelAndView();
        mv.addObject("data", data);
        mv.setViewName(error_path);
        return mv;
    }
    //处理请求参数格式错误 @RequestBody上validate失败后抛出的异常是MethodArgumentNotValidException异常。
    @ExceptionHandler(MethodArgumentNotValidException.class)
    public ModelAndView exceptionHander(HttpServletRequest request, MethodArgumentNotValidException e) throws Exception {
        String message = e.getBindingResult().getAllErrors().stream().map(DefaultMessageSourceResolvable::getDefaultMessage).collect(Collectors.joining());
        logger.error("路由 : {},RequestBody-Valid错误信息 : {}", request.getRequestURL(), message);
		//Data - 我定义的通用数据模型 
        Data<String> data = new Data(message, false, request.getRequestURL());

        ModelAndView mv = new ModelAndView();
        mv.addObject("data", data);
        mv.setViewName(error_path);
        return mv;
    }
    //处理请求参数格式错误 @RequestParam上validate失败后抛出的异常是javax.validation.ConstraintViolationException
    @ExceptionHandler(ConstraintViolationException.class)
    public ModelAndView exceptionHander(HttpServletRequest request, ConstraintViolationException e) throws Exception {
        String message = e.getConstraintViolations().stream().map(ConstraintViolation::getMessage).collect(Collectors.joining());
        logger.error("路由 : {},RequestParam-Valid错误信息 : {}", request.getRequestURL(), message);
		//Data - 我定义的通用数据模型 
        Data<String> data = new Data(message, false, request.getRequestURL());

        ModelAndView mv = new ModelAndView();
        mv.addObject("data", data);
        mv.setViewName(error_path);
        return mv;
    }
    //处理Get请求中 使用@Valid 验证路径中请求实体校验失败后抛出的异常,
    @ExceptionHandler(BindException.class)
    public ModelAndView exceptionHander(HttpServletRequest request, BindException e) throws Exception {
        String message = e.getBindingResult().getAllErrors().stream().map(DefaultMessageSourceResolvable::getDefaultMessage).collect(Collectors.joining());
        logger.error("路由 : {},Get-Valid错误信息 : {}", request.getRequestURL(), message);
		//Data - 我定义的通用数据模型 
        Data<String> data = new Data(message, false, request.getRequestURL());

        ModelAndView mv = new ModelAndView();
        mv.addObject("data", data);
        mv.setViewName(error_path);
        return mv;
    }
    //其它异常
    @ExceptionHandler(Exception.class)
    public ModelAndView exceptionHander(HttpServletRequest request, Exception e) throws Exception {
        logger.error("路由 : {},错误信息 : {}", request.getRequestURL(), e.getMessage());
		//Data - 我定义的通用数据模型 
        Data<String> data = new Data(e.getMessage(), false, request.getRequestURL());

        ModelAndView mv = new ModelAndView();
        mv.addObject("data", data);
        mv.setViewName(error_path);
        return mv;
    }
}
示例说明
  • 如下当我使用 @Valid 注解来使@Validated ,@Valid ,@NotBlank,@NotNull 等注解生效时,如果抛出异常将会被上面代码中的 MethodArgumentNotValidException 所拦截并处理
    在这里插入图片描述
  • 其他的不一一列举

Data - 我定义的通用数据模型 (高复用返回数据类)

import org.apache.commons.lang.StringUtils;
/**
 * @ ClassName: Reuse
 * @ Author: longxin
 * @ CreatTime: 2020/4/14 0014 15:41
 * @ version: 1.0
 */
@lombok.Data
public class Data<T> {
    private Integer code;
    private String msg;
    private Boolean success;
    private T data;

    public Data() {
    }
    public Data(T data) {
        this.data = data;
        reuseCheck();
    }
    public Data(String msg, T data) {
        this.data = data;
        this.msg = msg;
        reuseCheck();
    }
    public Data(String msg, Integer code, T data) {
        this.data = data;
        this.code = code;
        this.msg = msg;
        reuseCheck();
    }
    public Data(String msg, Boolean success, T data) {
        this.data = data;
        this.msg = msg;
        this.success = success;
        reuseCheck();
    }
    public Data(String msg, Integer code, Boolean success, T data) {
        this.data = data;
        this.code = code;
        this.msg = msg;
        this.success = success;
        reuseCheck();
    }
    private void reuseCheck(){
        msg = StringUtils.isBlank(msg) ? "操作成功." : msg;
        success = success == null ? true : success;
        code = code == null ? success ? 200 : 0 : code;
    }
}

BlogException 我定义的通用异常处理

import org.springframework.http.HttpStatus;

public class BlogException extends AbstractException {
	//状态
    private HttpStatus httpStatus;
    public BlogException(String message) {
        super(message);
    }
    public BlogException(String message, HttpStatus httpStatus) {
        super(message);
        this.httpStatus = httpStatus;
    }
    public BlogException(String message, Throwable cause) {
        super(message, cause);
    }
    @Override
    public HttpStatus getStatus() {
        return this.httpStatus == null ? HttpStatus.INTERNAL_SERVER_ERROR : this.httpStatus;
    }
    public Integer getCode(){
        return this.httpStatus.value();
    }
}

抽象异常类(BlogException 继承它)

import org.springframework.http.HttpStatus;
import org.springframework.lang.NonNull;
import org.springframework.lang.Nullable;

public abstract class AbstractException extends RuntimeException {
    private Object errorData;
    public AbstractException(String message) {
        super(message);
    }
    public AbstractException(String message, Throwable cause) {
        super(message, cause);
    }
    public abstract HttpStatus getStatus();
    @Nullable
    public Object getErrorData() {
        return errorData;
    }
    @NonNull
    public AbstractException setErrorData(@Nullable Object errorData) {
        this.errorData = errorData;
        return this;
    }
}

前端错误页接收代码

<!DOCTYPE html>
<html lang="en" xmlns:th="http://www.w3.org/1999/xhtml">
<body>
      <span th:text="${data}">整体</span>
      <span th:text="${data.code}">状态码</span>
      <span th:text="${data.msg}">信息</span>
      <span th:text="${data.data}">数据</span>
</body>
</html>

使用场景

代码中抛出

		try {
            //你的代码处
        }catch (Exception e){
            throw new BlogException("异常信息", "状态码");
        }

注解中生效

	@NotBlank(message = "用户名或邮箱不能为空")
    @Size(max = 255, message = "用户名或邮箱的字符长度不能超过 {max}")
    private String username;

其他系统异常:服务器异常,请求参数异常等等,很多地方都可以用到

使用实例截图

某Service登陆接口示例

登录时未查询到用户信息或密码不正确时抛出异常
在这里插入图片描述

测试效果

Blog错误:用户名或密码错误

  • 后台
    在这里插入图片描述
  • Swagger测试
    在这里插入图片描述

参数空异常

  • 后台
    在这里插入图片描述

  • Swagger测试
    在这里插入图片描述

  • 其他异常不一一列举

  • 0
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: springboot+thymeleaf项目是一种基于Java语言开发的Web应用程序。它采用了Spring Boot框架和Thymeleaf模板引擎,可以快速地搭建一个高效、可靠、易于维护的Web应用程序。该项目具有以下特点: 1. 简单易用:Spring Boot框架提供了一系列的自动化配置,使得开发者可以快速地搭建一个Web应用程序,而不需要过多的配置。 2. 高效可靠:Spring Boot框架采用了一系列的优化措施,使得应用程序具有高效、可靠的性能。 3. 易于维护:Thymeleaf模板引擎提供了一种简单、易于维护的模板语言,使得开发者可以快速地开发出具有良好可读性的Web应用程序。 总之,springboot+thymeleaf项目是一种非常优秀的Web应用程序开发框架,可以帮助开发者快速地开发出高效、可靠、易于维护的Web应用程序。 ### 回答2: Spring Boot是一个基于Spring框架的快速开发框架,这个框架的优点在于其简单易用,能够快速搭建一个Java Web应用程序,无需进行复杂的配置和繁琐的XML文件编写。而Thymeleaf则是一种Web和HTML的模板引擎,可以方便地处理文本、链接和表单等元素,支持多重继承和页面片段的复用等特性。 Spring BootThymeleaf的结合,可以帮助开发人员更加简便地搭建Web应用程序。在使用Spring Boot进行项目开发时,可以使用Thymeleaf来完成Web开发的视图层,进行模版板的渲染和数据绑定。这样就可以很直接地将数据通过模板引擎展现出来,且更加方便。 在一个Spring Boot Thymeleaf项目的构建中,需要进行如下步骤: 1. 首先,引入Spring BootThymeleaf的依赖以及其他必要的依赖,例如web和mybatis等相关组件。 2. 创建一个Controller类,并使用@Controller注解将类标记为Controller,编写具体的Action方法,这些方法可以用@RequestMapping或@GetMapping等注解来定义处理请求的URL路径和请求类型等相关信息。 3. 创建一个Model类,用于封装需要传输到前端的数据和相关操作等。 4. 在Controller内部设置Model变量并将相关数据注入Model,然后将需要展现的数据作为参数传递给Thymeleaf进行渲染,最后将渲染完成后的结果返回给前端页面展现。 5. 编写HTML页面,使用Thymeleaf标签来渲染动态数据。 需要注意的是,在进行Thymeleaf模板的渲染时,需要遵守一定的规范,例如页面中的数据变量名称需与Model中的属性名称一致,引入Thymeleaf命名空间等等。 总之,Spring BootThymeleaf结合使用可以帮助开发人员快速地完成Web开发,整个过程简单而且高效。使用Thymeleaf能够降低模版制作的门槛,进一步提高开发效率,并且能够提供丰富的模版处理标签,使得页面制作更加灵活。 ### 回答3: 近年来,使用SpringBootThymeleaf进行Web开发已经成为越来越多的开发者选择的方案。SpringBoot是一个基于Spring框架的快速Web应用开发框架,而Thymeleaf是一种基于HTML的模板引擎,其中需要了解的内容包括以下几点: 首先,SpringBoot框架的优点是非常明显的。它提供了很多便于使用的方法,例如自动装配,以及基于配置的许多默认值。这使得开发者可以花更少的时间和精力来开发项目,将重点放在业务逻辑和功能实现上。 其次,Thymeleaf是一种非常强大和灵活的模板引擎,其语法简单易懂,而且支持HTML5标准。它还提供了一些样式和布局的工具,以及易于使用的表达式和标签,使得Web页面开发更加容易。 当然,SpringBoot集成Thymeleaf的过程也并不复杂。只需添加thymeleaf-starter包依赖,SpringBoot将自动将Thymeleaf注册为默认的模板引擎。然后,您只需要编写Thymeleaf模板文件即可。 最后,值得注意的是,使用SpringBootThymeleaf进行Web开发的好处在于它们之间的紧密集成。这种紧密集成可以更轻松地创建动态和交互性的Web应用程序,这是传统的HTML和JavaScript不能提供的。 总的来说,SpringBootThymeleaf是一对非常强大且易于使用的Web开发工具组合,它们的出现大大提高了Web开发的效率和质量,同时也为开发人员提供了更好的开发体验。我们相信,这对于Web开发者来说是非常有价值的组合。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值