一文搞懂SpringMVC中@InitBinder注解的使用

一文搞懂SpringMVC中@InitBinder注解的使用

目录

简介

应用示例

原理解读

环境:Springboot2.4.12

简介

​@Controller或@ControllerAdvice类可以有@InitBinder方法来初始化WebDataBinder的实例,这些方法可以:

将请求参数(即表单或查询数据)绑定到模型对象。

将基于字符串的请求值(如请求参数、路径变量、头、cookie等)转换为控制器方法参数的目标类型。

渲染HTML表单时,将模型对象的值格式化为字符串值。

@InitBinder方法可以注册控制器特定的java.bean.PropertyEditor或Spring Converter和 Formatter组件。另外,你可以使用MVC配置在全局共享的FormattingConversionService中注册Converter和Formatter类型。

@InitBinder方法支持许多与@RequestMapping方法相同的参数,除了@ModelAttribute(命令对象)参数。通常,它们是用WebDataBinder参数(用于注册)和一个void返回值声明的。

应用示例 @RestController @RequestMapping("/demos") public class DemoController { @InitBinder // 1 public void bind(WebDataBinder binder) { // 2 binder.registerCustomEditor(Long.class, new PropertyEditorSupport() { // 3 @Override public void setAsText(String text) throws IllegalArgumentException { setValue(Long.valueOf(text) + 666L) ; } }) ; } @GetMapping("/index") public Object index(Long id) { return "index - " + id ; } }

注意以下几点:

使用 @InitBinder 注解。

接收 WebDataBinder 参数。

注册自定义的转换器。

方法返回值必须是 void。

在上面的示例中注册了一个类型转换器从字符串转换为Long类型 并且在原来值基础上增加了666L。

原理解读

HandlerAdapter 执行。

public class RequestMappingHandlerAdapter extends AbstractHandlerMethodAdapter implements BeanFactoryAware, InitializingBean { protected ModelAndView invokeHandlerMethod(HttpServletRequest request, HttpServletResponse response, HandlerMethod handlerMethod) throws Exception { // ... // 这里会查找当前执行的Controller中定义的所有@InitBinder注解的方法 WebDataBinderFactory binderFactory = getDataBinderFactory(handlerMethod); ServletInvocableHandlerMethod invocableMethod = createInvocableHandlerMethod(handlerMethod); invocableMethod.invokeAndHandle(webRequest, mavContainer); // ... } }

ServletInvocableHandlerMethod 执行。

public class ServletInvocableHandlerMethod extends InvocableHandlerMethod { public void invokeAndHandle(ServletWebRequest webRequest, ModelAndViewContainer mavContainer, Object... providedArgs) throws Exception { // 调用父类方法 Object returnValue = invokeForRequest(webRequest, mavContainer, providedArgs); // ... } } // 执行父类方法调用 public class InvocableHandlerMethod extends HandlerMethod { public Object invokeForRequest(NativeWebRequest request, @Nullable ModelAndViewContainer mavContainer, Object... providedArgs) throws Exception { Object[] args = getMethodArgumentValues(request, mavContainer, providedArgs); return doInvoke(args); } protected Object[] getMethodArgumentValues(NativeWebRequest request, @Nullable ModelAndViewContainer mavContainer, Object... providedArgs) throws Exception { Object[] args = new Object[parameters.length]; for (int i = 0; i < parameters.length; i++) { // 解析参数 args[i] = this.resolvers.resolveArgument(parameter, mavContainer, request, this.dataBinderFactory); } } }

参数解析

在上面的Controller示例中,参数的解析器是RequestParamMethodArgumentResolver。

调用父类的resolveArgument方法。

public abstract class AbstractNamedValueMethodArgumentResolver { public final Object resolveArgument(MethodParameter parameter, @Nullable ModelAndViewContainer mavContainer, NativeWebRequest webRequest, @Nullable WebDataBinderFactory binderFactory) throws Exception { // 封装方法参数的名称这里为:id NamedValueInfo namedValueInfo = getNamedValueInfo(parameter); // resolvedName = id Object resolvedName = resolveEmbeddedValuesAndExpressions(namedValueInfo.name); // ... // 获取参数名对应的请求参数值:/demos/index?id=100 , 这就返回100 Object arg = resolveName(resolvedName.toString(), nestedParameter, webRequest); // ... if (binderFactory != null) { // 根据当前的Request对象及请求参数名创建WebDataBinder对象 // 内部创建的ExtendedServletRequestDataBinder对象 WebDataBinder binder = binderFactory.createBinder(webRequest, null, namedValueInfo.name); try { // 执行类型转换 arg = binder.convertIfNecessary(arg, parameter.getParameterType(), parameter); } } } } // 创建WebDataBinder对象 public class DefaultDataBinderFactory implements WebDataBinderFactory { public final WebDataBinder createBinder(NativeWebRequest webRequest, @Nullable Object target, String objectName) throws Exception { WebDataBinder dataBinder = createBinderInstance(target, objectName, webRequest); if (this.initializer != null) { // 初始化WebDataBinder对象,这里最主要的就是为其设置类型转换器 this.initializer.initBinder(dataBinder, webRequest); } // 初始化执行@InitBinder注解的方法 initBinder(dataBinder, webRequest); return dataBinder; } } public class InitBinderDataBinderFactory extends DefaultDataBinderFactory { public void initBinder(WebDataBinder dataBinder, NativeWebRequest request) throws Exception { // 遍历所有@InitBinder注解的方法 for (InvocableHandlerMethod binderMethod : this.binderMethods) { if (isBinderMethodApplicable(binderMethod, dataBinder)) { // 这里就是执行@InitBinder注解的方法 Object returnValue = binderMethod.invokeForRequest(request, null, dataBinder); // 如果@InitBinder注解的方法有返回值则抛出异常 if (returnValue != null) { throw new IllegalStateException("@InitBinder methods must not return a value (should be void): " + binderMethod); } } } } } // 解析@InitBinder注解方法的参数及方法执行 public class InvocableHandlerMethod extends HandlerMethod { public Object invokeForRequest(NativeWebRequest request, @Nullable ModelAndViewContainer mavContainer, Object... providedArgs) throws Exception { // 解析获取@InitBinder注解方法的参数 Object[] args = getMethodArgumentValues(request, mavContainer, providedArgs); // 执行调用 return doInvoke(args); } }

执行类型转换

在上面执行流程中,我们知道获取了一个WebDataBinder对象和由@InitBinder 注解的方法的调用执行。接下来就是进行类型的转换。

public abstract class AbstractNamedValueMethodArgumentResolver { public final Object resolveArgument(MethodParameter parameter, @Nullable ModelAndViewContainer mavContainer, NativeWebRequest webRequest, @Nullable WebDataBinderFactory binderFactory) throws Exception { if (binderFactory != null) { // 根据当前的Request对象及请求参数名创建WebDataBinder对象 // 内部创建的ExtendedServletRequestDataBinder对象 WebDataBinder binder = binderFactory.createBinder(webRequest, null, namedValueInfo.name); try { // 执行类型转换 arg = binder.convertIfNecessary(arg, parameter.getParameterType(), parameter); } } } } // 最终通过该类调用类型转换 class TypeConverterDelegate { public <T> T convertIfNecessary(@Nullable String propertyName, @Nullable Object oldValue, @Nullable Object newValue, @Nullable Class<T> requiredType, @Nullable TypeDescriptor typeDescriptor) throws IllegalArgumentException { // Custom editor for this type? // 获取自定义的类型转换器(首先获取的就是我们上面自定义的) PropertyEditor editor = this.propertyEditorRegistry.findCustomEditor(requiredType, propertyName); // ... Object convertedValue = newValue; // ... convertedValue = doConvertValue(oldValue, convertedValue, requiredType, editor); } private Object doConvertValue(@Nullable Object oldValue, @Nullable Object newValue, @Nullable Class<?> requiredType, @Nullable PropertyEditor editor) { // ... if (convertedValue instanceof String) { if (editor != null) { String newTextValue = (String) convertedValue; // 最终的调用 return doConvertTextValue(oldValue, newTextValue, editor); } else if (String.class == requiredType) { returnValue = convertedValue; } } return returnValue; } // 最终得到了我们想要的值 private Object doConvertTextValue(@Nullable Object oldValue, String newTextValue, PropertyEditor editor) { try { editor.setValue(oldValue); } // ... editor.setAsText(newTextValue); return editor.getValue(); } }

以上就是参数绑定及类型转换的过程。

到此这篇关于一文搞懂SpringMVC中@InitBinder注解的使用的文章就介绍到这了,更多相关SpringMVC @InitBinder注解内容请搜索易知道(ezd.cc)以前的文章或继续浏览下面的相关文章希望大家以后多多支持易知道(ezd.cc)!

推荐阅读

    字符库快捷键|字符串快捷键

    字符库快捷键|字符串快捷键,,1. 字符串快捷键1、单行注释单行注释是 #Mac的快捷键是 command+/windows的快捷键是 Ctrl + /2、多行注

    做模型电脑|电脑上做模型的软件

    做模型电脑|电脑上做模型的软件,,电脑上做模型的软件建模软件 中文名 ;sketchup渲染软件 主要用在建筑方案中:对建筑造型的推敲 现在G

    Python之可迭代对象、迭代器、生成器

    Python之可迭代对象、迭代器、生成器,迭代,生成器,一、概念描述可迭代对象就是可以迭代的对象,我们可以通过内置的iter函数获取其迭代器,可

    应用程序对象

    应用程序对象,,应用程序对象是一个应用程序级对象,用于在所有用户之间共享信息,并且在Web应用程序运行期间可以保存数据。 应用的性质: 方法

    为什么WPS表单总是不打开

    为什么WPS表单总是不打开,,为什么WPS表格总是打不开??百度知道为您找到了6条优质回答span>a,type:cluster">很有可能文件已经损坏。恢复

    电脑显示器模型定制|曲面显示器建模

    电脑显示器模型定制|曲面显示器建模,,曲面显示器建模可以,但效果还是平面的好。曲面屏显示器的优点:1.更广的视野(在同尺寸下,视角效果更广,视

    laravel-admin|自定义表单与验证

    laravel-admin|自定义表单与验证,表单,场景,场景:很多时候,由于我们业务场景比较特殊,需要自定义表单,然后框架给我提供了对应表单组建!案列:以

    Java创建对象的几种方式

    Java创建对象的几种方式,对象,方法,本文目录Java创建对象的几种方式java中几种创建对象的方式1Java中创建对象的集中方式有那些JAVA创建对