首页>资讯>正文
每日热讯!Spring MVC中@InitBinder注解是如何应用的?
2023-05-08 17:19:11    来源:清一色财经

@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。

原理解读

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 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();  }}

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

关键词:

每日热讯!Spring MVC中@InitBinder注解是如何应用的?

@InitBinder方法可以注册控制器特定的java bean PropertyEditor或SpringConverter和Formatter组件。2023-05-08

天天微头条丨千方百计助高校毕业生就业

高校毕业生就业工作事关高质量发展、民生福祉和社会稳定。今年春季开学以来,教育部会同有关部门推动加快升2023-05-08

速看:国家医保局:一季度基本医疗保险基金整体运行平稳

国家医保局日前公布了2023年一季度我国基本医疗保险和生育保险运行情况,基本医保基金总收入约9100亿元,总2023-05-08

《塞尔达传说王国之泪》第二座神庙怎么过?第二座神庙通关攻略

塞尔达传说王国之泪游戏中第二座神庙怎么过?可能很多玩家还不太清楚。下面小编给大家带来塞尔达传说王国之2023-05-08

男子火车上被杀 有人喊住手被凶手追 基本信息讲解_全球快播

大家好,今日关于【男子火车上被杀有人喊住手被凶手追】的话题登上了各大平台的热搜榜,受到全网的关注度非2023-05-08

关于龚琳娜哄骗美依礼芽吃折耳根及龚琳娜哄骗美依礼芽吃折耳根详情

音频解说1、龚琳娜哄骗美依礼芽吃折耳根,美依礼芽吃完以后整个人表情都变了,网友直呼“表情太可爱”。2、2023-05-08

每日视点!海南周刊 | 古村古名有典故

原标题:海南周刊|古村古名有典故开栏的话地名如人名,与生于斯长于斯的一代又一代人息息相关,见证着一个2023-05-08

天天快消息!银保监会:原则同意华泰汽车金融进入破产程序

财经网汽车5月8日讯,银保监会官网发布的批复显示,湖南三湘银行申请华泰汽车金融有限公司破产清算的请示获2023-05-08

郾城区孟庙镇三措并举喜迎“食博会”_天天速读

近日,孟庙镇通过环境卫生整治、做好安全生产和矛盾纠纷大排查三措并举,喜迎第二十届食博会召开。一是加大2023-05-08

大众id.7谍照曝光,欧洲市场将取代帕萨特轿车定位-天天信息

日前,外媒曝光了一组大众ID 7Space的谍照,新车是ID 7的旅行版,最快会在今年底正式发布。在欧洲市场,ID2023-05-08

感恩作文结尾_都是怎么总结的-焦点观察

欢迎观看本篇文章,小升来为大家解答以上问题。感恩作文结尾,都是怎么总结的很多人还不知道,现在让我们一2023-05-08

今日最新!入会仪式上,专门给孩子们留一方舞台

入会仪式上,专门给孩子们留一方舞台_浙江工人日报网 ">入会仪式上,专门给孩子们留一方舞台-浙江工人日报2023-05-08

景德镇警方打掉一个吸贩毒团伙 抓获32名涉毒人员 全球快播报

5月5日,景德镇市公安局民警深挖线索,主动出击,分别将涉毒犯罪嫌疑人艾某、占某抓获归案。经查,艾某对其2023-05-08

2022冬奥会手抄报_世界热闻

1、第24届冬奥会于2022年02月04日至02月20日在中国北京和张家口举行。2、冬季奥林匹克运动会(Olympic2023-05-08

焦点资讯:尔康制药:5月5日融资买入181.59万元,融资融券余额2.07亿元

5月5日,尔康制药(300267)融资买入181 59万元,融资偿还410 06万元,融资净卖出228 47万元,融资余额2 07亿元。2023-05-08

世界微动态丨火车上被害男子亲属:错过抢救时间 死者与凶手不认识

据媒体报道,5月4日在K435次列车上一男子被人持刀杀害,凶手当场被抓获。记者联系死者表哥李先生,他表示:2023-05-08

全球热消息:土耳其拒绝向乌军提供S-400导弹系统

土耳其外长恰武什奥卢5月7日表示,土耳其方面拒绝了美方提出的、向乌克兰提供俄制S-400防空导弹系统使用权2023-05-08

全球消息!期货股票投资的魅力(二)

文 上甲资讯 李哥聊期货888交易领域能让交易员在一个更快的节奏下,直接真实的体2023-05-08

当前聚焦:科慕携手公众,发现化学之美,催化美好未来

上海2023年5月6日 美通社 --科慕公司(简称 "科慕 ",纽交所代码:CC)是一家跨国化学公司,在钛白科技、热2023-05-08

全球新资讯:中国式现代化的长三角实践 | 金桥5G产业生态园产业集聚效应助力企业发展

6日,“中国式现代化的长三角实践”网络主题采访团一行走进位于上海浦东新区的金桥5G产业生态园,通过实地2023-05-08

热资讯!延边东北亚技术产权交易有限公司

1、延边东北亚技术产权交易有限公司于2016年04月01日成立。2、法定代表人金炳哲,公司经营范围包括:知识产2023-05-08

每日观点:中国药学杂志好投吗_中国药学杂志

1、是的参考资料:中文核心期刊要目总览第五编医药、卫生R9--药学:药学学报2、中国药学杂志3、中国医院药2023-05-08

乡科级领导职务及相当职务层次_职务层次 环球焦点

1、评审工程师、经济师(中级职称)的条件是:大学本科毕业后,担任助理级职务满三年以上;专科毕业后,担任2023-05-08

荣辱不惊闲看庭前花开花落对联_荣辱不惊闲看庭前花开花落-天天微头条

1、这句话的意思是说,为人做事能视宠辱如花开花落般平常,才能不惊;视职位去留如云卷云舒般变幻。2、才能2023-05-08

西宁市湟中区土门关乡 “党建+”模式激活基层治理“神经末梢” 环球观焦点

近年来,西宁市湟中区土门关乡坚持以党建引领基层治理,深化“党建+”规范化建设,进一步畅通乡村治理“毛2023-05-08

滚动
资讯