当前位置:   article > 正文

爆破专栏丨SpringBoot2.x系列教程之利用InitBinder处理请求参数的绑定(一)_springboot initibinder

springboot initibinder

前言

我们在开发项目时,前后端之间需要密切配合,才能确保项目的稳定高效。前端通过URL接口给后端传递请求参数,后端根据这些不同的请求参数返回对应的响应信息。那么在这个传参和返回响应信息的过程中,会发生什么事情呢?我们能不能在这个过程中做一些干预操作呢?比如我们要添加一个学生的年龄信息,前端传递过来一个负数作为年龄参数,这个明显不合法,那么后端能不能对这个非法的参数做出一些处理呢?

所以今天 壹哥 就带大家在SpringBoot中,学习一些可以处理请求参数的技巧,确保我们的后端接口更加的安全健壮。

一. SpringMVC请求参数绑定流程

1. 请求参数绑定流程

我们在开发的时候,经常会从html,jsp等页面中将请求参数通过request对象传递到后端。可是经常会遇到这么一种情况,那就是从前端传递过来的数据参数到达后端后,还需要再把该数据参数进行适当的抽取、转换、校验等操作,进而组装成另一种格式的对象。这个转换的工作,可以由SpringMVC提供的@InitBinder注解来完成。

图片

2. SpringMVC中请求参数的绑定

SpringMVC可以自动将request中的请求参数绑定到对象的每个property属性上,但只会把一些简单的数据类型(比如String, int, float等)绑定到对应的属性上。可是如果面对复杂对象,那就要借助PropertyEditor这个接口来完成数据的绑定了。

PropertyEditor这个接口提供了两个方法,一个方法是将String类型的值转成property对应的数据类型,另一个方法是将一个property属性转成String,如下图所示:

图片

3. CustomDateEditor继承关系

接下来我会结合CustomDateEditor与InitBinder,给大家讲解如何对前端传递过来的日期类型参数进行格式化校验。我们先来看看CustomDateEditor类的结构关系,如下图所示:

图片

4. 示例代码

这里我先放一段校验前端日期类型参数格式的示例代码,后面我们会利用这段代码实现对前端传入的日期参数校验的目的。

  1. @InitBinder
  2. public void InitBinder(WebDataBinder binder) {
  3.     //前端传入的时间格式必须是"yyyy-MM-dd"效果!
  4.     DateFormat df = new SimpleDateFormat("yyyy-MM-dd");
  5.     CustomDateEditor dateEditor = new CustomDateEditor(df, true);
  6.     binder.registerCustomEditor(Date.class, dateEditor);
  7. }

二. @InitBinder详解

在开始今天的内容之前,我先给大家介绍一个@InitBinder注解。

1. @InitBinder注解简介

@InitBinder注解可以作用在@Controller的方法上,表示为当前控制器注册一个属性编辑器,对WebDataBinder进行初始化,且只对当前的Controller有效。

2. @InitBinder执行时机

@InitBinder注解被解析的时机,是在其所位于的方法被请求执行之前。同时@InitBinder标注的方法是可以多次执行的,也就是说来一次请求就会执行一次@InitBinder解析。

3. @InitBinder执行原理

当某个Controller上的第一次请求,由SpringMVC前端控制器匹配到该Controller之后,根据Controller的 class类型来查找所有标注了@InitBinder注解的方法,并且存入Request Mapping Handler Adapter里的 initBinderCache 缓存中。等下一次请求执行对应业务方法之前,会先走initBinderCache缓存,而不用再去解析@InitBinder。

三. @InitBinder实现过程

接下来我们编写一个案例,实现对前端传递过来的日期参数进行格式化校验。

首先我们来创建一个Web程序,创建过程请参考之前的案例,此处略!

1. 创建Controller测试接口

在创建的Web项目中,我创建一个Controller接口,接口方法中可以接受Date类型的参数。

  1. package com.yyg.boot.web;
  2. import lombok.extern.slf4j.Slf4j;
  3. import org.springframework.beans.propertyeditors.CustomDateEditor;
  4. import org.springframework.beans.propertyeditors.StringTrimmerEditor;
  5. import org.springframework.stereotype.Component;
  6. import org.springframework.web.bind.WebDataBinder;
  7. import org.springframework.web.bind.annotation.*;
  8. import org.springframework.web.bind.support.WebBindingInitializer;
  9. import org.springframework.web.context.request.WebRequest;
  10. import java.text.DateFormat;
  11. import java.text.ParseException;
  12. import java.text.SimpleDateFormat;
  13. import java.util.Date;
  14. import java.util.HashMap;
  15. import java.util.Map;
  16. /**
  17.  * @Description Description
  18.  * @Author 一一哥Sun
  19.  * @Date Created in 2020/3/23
  20.  */
  21. @Slf4j
  22. @RestController
  23. public class BindController {
  24.     @GetMapping(value = "/bind")
  25.     public Map<StringObject> getFormatData(Date date) throws ParseException {
  26.         log.warn("date={}"date);
  27.         Map<StringObject> map = new HashMap<>();
  28.         map.put("name""一一哥");
  29.         map.put("age"30);
  30.         map.put("date"date);
  31.         return map;
  32.     }
  33. }

项目入口类的创建这里我也直接略过了,大家自行创建即可。

2. 启动程序进行测试

启动项目后,此时我们可以在Postman中输入如下地址进行测试:http://localhost:8080/bind?date=2020-09-09

Postman中的效果如下所示:

图片

经过测试,发现此时产生400状态码,我们知道产生400状态码的原因是前后端参数不一致,具体就是无法将前端传递过来的String类型的时间字符串转换为Date类型!因为前端是没有Date类型的,前端传递的日期参数只能被理解为String类型,默认情况下是不能把这个String类型的日期数据直接转换为Date类型的。

图片

那怎么解决这个问题呢?

3. 添加@InitBinder代码

接下来我们在上面创建的Controller里面,添加一段新的代码,如下:

  1. /**
  2.      * @InitBinder标注的方法,只针对当前Controller有效!
  3.      * 如果没有该方法,则会产生400状态码!
  4.      * MethodArgumentTypeMismatchException: Failed to convert value of type 'java.lang.String' to required type 'java.util.Date!
  5.      */
  6.     @InitBinder
  7.     public void InitBinder(WebDataBinder binder) {
  8.         //前端传入的时间格式必须是"yyyy-MM-dd"效果!
  9.         DateFormat df = new SimpleDateFormat("yyyy-MM-dd");
  10.         CustomDateEditor dateEditor = new CustomDateEditor(df, true);
  11.         binder.registerCustomEditor(Date.class, dateEditor);
  12.     }

此时完整的Controller类如下图所示:

图片

4. 重启项目进行测试

添加完代码之后,我们把项目重启,然后我们在Postman中重新输入如下地址:http://localhost:8080/bind?date=2020-09-09

Postman的测试效果如下:

图片

可以发现此时前端传递的时间字符串被成功的传递到后端,并且被成功的转换成了Date类型!以上就是@InitBinder的原理及用法!

注意:@InitBinder属于Controller级别的属性编辑器,并不是全局级别(针对所有@Controller)的属性编辑器,也就是一个@InitBinder只对当前所在的Controller有效,对其他Controller是无效的!那如果我们的项目中,有多个Controller中都要进行日期格式转换怎么办呢?请先思考一下。

5. 项目结构图

最终代码结构如下图所示,各位可以参考创建。

图片

需要更多教程,微信扫码即可

 

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/我家自动化/article/detail/66288
推荐阅读
相关标签