@InitBinder方法可以注册控制器特定的java.bean.PropertyEditor或Spring Converter和 Formatter组件。
【资料图】
环境: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。
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 { publicT convertIfNecessary(@Nullable String propertyName, @Nullable Object oldValue, @Nullable Object newValue, @Nullable Class 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(); }}
以上就是参数绑定及类型转换的过程。
@InitBinder方法可以注册控制器特定的java bean PropertyEditor或...
卫光生物(002880)05月08日在投资者关系平台上答复了投资者关心的问题。
北京医保报销2023年新规是什么?社保网小编为您整理了最新资讯。202...
为全面深化我区“一网通办”改革工作,进一步优化企业和群众办事流...
大家好,小评来为大家解答以上问题。微pe安装原版win7系统,微pe安...
上证报中国证券网讯据中国物流与采购联合会5月8日消息,中国物流与...
如果面临的是决赛,那么葡萄牙人打封闭也会上场,但这是180分钟的较...
2023青海特岗教师招聘报考条件包括:1 中国国籍遵纪守法、热爱教育...
业内认为,我国现有的氢能标准单一、笼统、松垮、割裂等问题值得警...
1、“怪”字开头的成语有怪音、怪貌、怪奇、怪雨、瞎风等。2、1 怪...
走进乌鲁木齐市米东区揽胜街,这里绿化错落有致,公交站台简约现代...
豆瓣8 9,这部奇幻设定的现代版“聊斋”,太治愈了!,喜剧,豆瓣,治...
韭菜片片,也叫韭菜面片,是一种乡下小吃,街头很少有卖的,自己做...
7点早安最近天气越来越热走在街上总是能看见好多好看的小姐姐都穿上...
推动外储规模上升的主要因素仍是汇率变动和资产价格变化。温彬同时...
中国石油网消息(记者宋鹏通讯员管晶李鹏飞)新疆油田公司玛湖油区...
交通物流是经济循环“大动脉”。中国物流与采购联合会副会长任豪祥5...
江西新闻客户端讯(通讯员邱志超)5月6日5时许,乐安县公安局龚坊派...
5月7日,国际篮联三人女篮系列赛武汉站的比赛全部结束,中国女篮以...
【张劲已被刑事控制!经侦调查收网雪松系公司被立案查处涉嫌“非吸...
1、这个哪有固定的呢。2、看需要吧。3、需要放视频。4、那就需要视...
今天来聊聊关于金骏眉怎样冲泡,金骏眉怎么泡的文章,现在就为大家...
“我个人的建议是,不要投资这个芯片项目”。今天下午,面对投资人...
中建交通第二建设公司近日举办了“学习二十大我与企业共成长”商务...
咸安区卫生健康系统第一个基层红十字组织成立---5月6日,在代表们热...
据中超联赛官方消息,上海申花主教练吴金贵当选中超4月最佳主帅。官...
一季度同比增9 5%武汉乘用车单月销量首破5万辆---湖北日报讯(记者...
5月6日晚,在第十九届中国中学生排球联赛(高中组)决赛中,黑龙江...
1、millionnum 1 百万;百万个(人或物)n 1 百万元;百万(镑,美元,法郎)2 无数,许许
近日,在微博等平台上一条短视频冲上热搜某女演员发布视频称自己孕...