SpringBoot2.x系列教程29--整合SpringMVC之@InitBinder处理请求参数的绑定(一)

SpringBoot2.x系列教程29--整合SpringMVC之@InitBinder处理请求参数的绑定(一)

作者:一一哥

一.Spring请求参数绑定流程

1.请求参数绑定流程

我们在开发的时候,经常会从html,jsp中将请求参数通过request对象传递到后台。可是经常会遇到这么一种情况,那就是传过来的数据到后台后,还要再组装成一种对象的格式。这时候Spring提供的@InitBinder注解就发挥了很大的作用。

SpringBoot2.x系列教程29--整合SpringMVC之@InitBinder处理请求参数的绑定(一)_第1张图片

2.Spring中请求参数绑定

Spring可以自动将request中的请求参数数据绑定到对象的每个property上,但是只会绑定一些简单数据类型(比如Strings, int, float等)到对应的对象中。可是如果面对复杂的对象,那就要借助PropertyEditor
来帮助我们完成复杂对象的绑定。

PropertyEditor这个接口提供了两个方法,一个是方法是将String类型的值转成property对应的数据类型,另一个方法是将一个property转成String。

SpringBoot2.x系列教程29--整合SpringMVC之@InitBinder处理请求参数的绑定(一)_第2张图片

3.CustomDateEditor继承关系

SpringBoot2.x系列教程29--整合SpringMVC之@InitBinder处理请求参数的绑定(一)_第3张图片

4.示例代码

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

二.@InitBinder详解

1.@InitBinder注解简介

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

2. @InitBinder执行时机

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

3. @InitBinder执行原理

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

三. @InitBinder实现过程

我们先创建一个web程序,过程请参考之前的案例,过程略!

项目结构图:

SpringBoot2.x系列教程29--整合SpringMVC之@InitBinder处理请求参数的绑定(一)_第4张图片

1. 创建Controller测试接口

package com.yyg.boot.web;

import lombok.extern.slf4j.Slf4j;
import org.springframework.beans.propertyeditors.CustomDateEditor;
import org.springframework.beans.propertyeditors.StringTrimmerEditor;
import org.springframework.stereotype.Component;
import org.springframework.web.bind.WebDataBinder;
import org.springframework.web.bind.annotation.*;
import org.springframework.web.bind.support.WebBindingInitializer;
import org.springframework.web.context.request.WebRequest;

import java.text.DateFormat;
import java.text.ParseException;
import java.text.SimpleDateFormat;
import java.util.Date;
import java.util.HashMap;
import java.util.Map;

/**
 * @Description Description
 * @Author 一一哥Sun
 * @Date Created in 2020/3/23
 */
@Slf4j
@RestController
public class BindController {

    @GetMapping(value = "/bind")
    public Map getFormatData(Date date) throws ParseException {
        log.warn("date={}", date);
        Map map = new HashMap<>();
        map.put("name", "一一哥");
        map.put("age", 30);
        map.put("date", date);
        return map;
    }

}

2. 启动程序进行测试

此时我们在postman中输入地址:
http://localhost:8080/bind?date=2020-09-09

SpringBoot2.x系列教程29--整合SpringMVC之@InitBinder处理请求参数的绑定(一)_第5张图片


经过测试,发现此时产生400状态码,具体原因是无法将前端传递过来的String类型的时间字符串转换为Date类型!

SpringBoot2.x系列教程29--整合SpringMVC之@InitBinder处理请求参数的绑定(一)_第6张图片

3. 添加@InitBinder代码,重新测试

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

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

SpringBoot2.x系列教程29--整合SpringMVC之@InitBinder处理请求参数的绑定(一)_第7张图片

然后我们在postman中重新输入地址:
http://localhost:8080/bind?date=2020-09-09

SpringBoot2.x系列教程29--整合SpringMVC之@InitBinder处理请求参数的绑定(一)_第8张图片


可以发现前端传递的时间字符串被成功的传递到后端,并且转换成了Date类型!

以上就是@InitBinder的原理及用法!

注意:

@InitBinder属于Controller级别的SpringMVC属性编辑器,并不是全局级别(针对所有@Controller)的属性编辑器哦!

你可能感兴趣的:(SpringBoot2.x系列教程29--整合SpringMVC之@InitBinder处理请求参数的绑定(一))