之前两篇文章 Spring-boot自定义参数校验注解如何在spring-boot中进行参数校验,我们介绍了,参数校验以及如何自定义参数校验注解,但是当传递参数出错时,只是把错误信息打印到了控制台,合理的做法是应该把校验的错误信息返回给前端,告知用户那里有问题,下面就这一步内容进行说明。

请求body参数

上篇文章 Spring-boot自定义参数校验注解的最后,在控制台打印了校验出错的信息

出错的异常类是MethodArgumentNotValidException,那如果想要自定义异常的返回,就需要在全局的异常处理器中针对这种异常进行处理。

在这篇文章 spring-boot自定义异常返回中,我们说了如何进行自定义异常的返回,参数校验的错误信息返回依然按照此方式进行处理,在全局异常处理类中定义异常处理方法:

@ExceptionHandler(value = MethodArgumentNotValidException.class)
@ResponseStatus(HttpStatus.BAD_REQUEST)
@ResponseBody
public UnifyResponse handlerBeanValidationException(HttpServletRequest request,
MethodArgumentNotValidException ex) {
String requestUri = request.getRequestURI();
String method = request.getMethod(); List<ObjectError> errors = ex.getBindingResult().getAllErrors();
return UnifyResponse.builder()
.code(5000)
.message(formatError(errors))
.requestUri(method + " " + requestUri)
.build();
} private String formatError(List<ObjectError> errors) {
StringBuilder builder = new StringBuilder();
errors.forEach(error -> builder.append(error.getDefaultMessage()).append(";"));
return builder.toString();
}

我们来对上面的代码进行一下解释:

  • 因为这个处理方法只是针对MethodArgumentNotValidException这个异常进行处理,所以@ExceptionHandler(value = MethodArgumentNotValidException.class)这里指定
  • @ResponseStatus(HttpStatus.BAD_REQUEST),所有的参数校验错误都是一类的,状态码设置为HttpStatus.BAD_REQUEST,也就是code等于400,当然也可以定义为其他的,按照自己业务需求定义就好,可以参考这篇文章 spring-boot自定义异常返回里关于自定义状态码的部分。
  • @ResponseBody,因为这个异常处理方法要返回自定义的对象,所以要使用这个注解,不然spring-boot是不会对自定义对象进行序列化的
  • List<ObjectError> errors = ex.getBindingResult().getAllErrors()进行参数校验的时候,可能多个参数都有问题,我们希望能够有问题的参数的错误信息全部都返回回去,所以这里要获取所有的错误。

回顾一下参数的定义,对这里有疑惑的同学可以看一下这篇文章Spring-boot自定义参数校验注解

@Builder
@Getter
@Setter
@PasswordEqual(min = 5, message = "密码和确认密码不一样")
public class UserDto { private int userId; @Length(min = 2, max = 10, message = "用户名长度必须在2-10的范围内")
private String username; private String password; private String confirmPassword;
}

接下来我们定再定义一个简单的接口,当传参出错时看异常处理方法能否按照定义的那样返回错误信息

@RequestMapping("/v2/user/create")
public UserDto createUser(@RequestBody @Validated UserDto userDto){
return userDto;
}

我们先来构造一个密码和确认密码不一致的情况

可以看到定义的错误信息被返回,而且状态码和自定义的code都是符合设计的,接下来我们再看一下多个参数错误的场景:

上面的场景中,用户名是不符合要求的,密码和确认密码也不一样,所以会产生两条错误信息,将其拼接到一起,返回给前端。

之前讨论的都是body里提交的参数,接下来我们看下路径参数或者查询参数校验出错时的处理

查询参数和路径参数

我们先定义两个接口一个是路径参数查询信息,一个是通过查询参数查询信息

@GetMapping("/v2/user/info")
public UserDto getUserInfo(@RequestParam @Length(min = 2, max = 5, message = "用户名长度必须在2-5的范围")
String username){
return UserDto.builder()
.userId(1000)
.username(username)
.build();
} @GetMapping("/v2/user/{username}")
public UserDto getUserInfoV2(@PathVariable @Length(min = 2, max = 5, message = "用户名长度必须在2-5的范围") String username){
return UserDto.builder()
.userId(2000)
.username(username)
.build();
}

然后我们访问这两接口,当发生错误时,看看他们会不会进入上文定义的异常处理方法中:

很明显,并没有进入上文定义的异常处理方法中,而是进入了handleException这个异常方法当中,这个算是个兜底的异常处理方法。

看一下控制台的输出:

这里抛出了ConstraintViolationException异常,这个异常我们并没有定制对应的异常处理函数,下面我们就来写一下:

@ExceptionHandler(ConstraintViolationException.class)
@ResponseStatus(HttpStatus.BAD_REQUEST)
@ResponseBody
public UnifyResponse handlerConstraintViolationException(HttpServletRequest request, ConstraintViolationException ex){
String requestUri = request.getRequestURI();
String method = request.getMethod(); Set<ConstraintViolation<?>> errors = ex.getConstraintViolations();
return UnifyResponse.builder()
.code(6000)
.message(formatConstraintException(errors))
.requestUri(method + " " + requestUri)
.build();
} private String formatConstraintException(Set<ConstraintViolation<?>> constraintViolations){
StringBuilder builder = new StringBuilder();
constraintViolations.forEach(constraintViolation -> builder.append(constraintViolation.getMessage()));
return builder.toString();
}

整体来说异常处理和上文几乎是一样的,只是获取错误message的方式不一样而已,我们再请求一下:

至此参数校验的错误message自定义返回,都完成了。

本文链接:https://www.immortalp.com/articles/2020/05/16/1589623786527.html

欢迎大家去 我的博客 瞅瞅,里面有更多关于测试实战的内容哦!!

前端:参数传错了,spring-boot:那错误信息我给你显示的友好点儿的更多相关文章

  1. 【spring boot】配置信息

    ======================================================================== 1.feign 超时配置 2.上传文件大小控制 3.J ...

  2. java spring boot 开启监控信息

    效果: 配置 // pom <dependency> <groupId>org.springframework.boot</groupId> <artifac ...

  3. Spring Boot自定义错误页面,Whitelabel Error Page处理方式

    我已经是Spring Framework框架的忠实粉丝.对于企业软件开发者来说它提供了对常见问题的通用解决方案,包括那些你在未来开发中没有意识到的问题.但是,它构建的J2EE项目变得比较臃肿,需要被一 ...

  4. (后端)Spring Boot自定义错误页面,Whitelabel Error Page处理方式(转)

    我已经是Spring Framework框架的忠实粉丝.对于企业软件开发者来说它提供了对常见问题的通用解决方案,包括那些你在未来开发中没有意识到的问题.但是,它构建的J2EE项目变得比较臃肿,需要被一 ...

  5. spring boot 启动错误:Could not resolve placeholder

    在启动整个spring boot项目时,出现错误: Could not resolve placeholder 原因:没有指定好配置文件,因为src/main/resources下有多个配置文件,例如 ...

  6. Spring Boot自定义错误视图

    Spring Boot缺省错误视图解析器 Web应用在处理请求的过程中发生错误是非常常见的情况,SpringBoot中为我们实现了一个错误视图解析器(DefaultErrorViewResolver) ...

  7. spring boot:thymeleaf给fragment传递参数的方法(spring boot 2.3.3)

    一,thymeleaf如何给fragment传递参数? 1,如果是全局的参数,可以用interceptor中传递 非全局参数,可以从controller中传递 2,引用片断时也可以传递参数 说明:刘宏 ...

  8. Spring boot 启动错误处理:Action: Consider the following: If you want an embedded database (H2, HSQL or Derby), please put it on the classpath. If you have database settings to be loaded from a particular...

    错误原因 在pom中引入了mybatis-spring-boot-starter ,Spring boot默认会加载org.springframework.boot.autoconfigure.jdb ...

  9. Spring Boot实现学生信息增删改查

    上一篇博客写了如何初始化一个简单的Spring Boot项目,这次详细记录一下如何连接数据库并实现增删改查基本操作. 我使用的是MySQL 5.5+Navicat,MySQL量级比较轻,当然微软的SQ ...

随机推荐

  1. Apache solr velocity模块 漏洞复现

    0x01 Solr简单介绍 Solr是建立在Apache Lucene ™之上的一个流行.快速.开放源代码的企业搜索平台. Solr具有高度的可靠性,可伸缩性和容错能力,可提供分布式索引,复制和负载平 ...

  2. 记使用STL与unique_ptr造成的事故-段子类比

    最近由于业务需要在写内存池子时遇到了一个doule-free的问题.折腾半个晚上以为自己的眼睛花了.开始以为是编译器有问题(我也是够自信的),但是在windows下使用qtcreator vs2017 ...

  3. 杂园日记-H5-IOS-Android混合开发

    1.js 调用 原生API iOS: window.webkit.messageHandlers.yourFunName.postMessage({"1":"3" ...

  4. serialize和json_encode 区别

    (1)serialize主要用于php的序列化,存储到文件或者数据库中,json_encode 也是序列化,但是 主要用于与其他语言比如js进行交互使用,对于传输来说,json有许多优点. (2)在显 ...

  5. [Linux] 检查是否已有进程在运行

    出处:sblim-sfcb-1.4.9 / sfcBroker.c int process_is_running() { #define STRBUF_LEN 512 #define BUF_LEN ...

  6. 在java中构建高效的结果缓存

    文章目录 使用HashMap 使用ConcurrentHashMap FutureTask 在java中构建高效的结果缓存 缓存是现代应用服务器中非常常用的组件.除了第三方缓存以外,我们通常也需要在j ...

  7. web 之 tomcat 8.5 和9.0如何进入manager?

    tomcat 8.5 和9.0如何进入manager? 第一步找到tomcat-user.xml文件 第二步添加如下代码 <role rolename="manager-gui&quo ...

  8. C6 C7的开机启动流程

    C6开机启动流程 1.内核引导,加电自检(通电后检查内核):检查bios的配置,检测硬件 装好系统之后才会进行以下内容 MBR 引导 (3.2.1...) GRUB菜单 (选择不同的系统)(按e,进入 ...

  9. 【linux题目】第三关

    1. 解释下什么是GPL,GNU,自由软件? 2. 如何选择Linux操作系统版本? 3. 安装系统时如何给Linux操作系统分区? 4. 描述Linux系统的启动过程? 5. 简要说出20个Linu ...

  10. Netflix:当你按下“播放”的时候发生了什么?

    从用户端来看,使用Netflix是很简单的,按下播放键之后视频就像变魔术一样完美呈现了.看起来很容易是吧?然而实际不是这样的.了解过云计算的人可能会简单地以为,既然Netflix使用AWS来提供视频服 ...