当前位置:网站首页>HandlerMethodArgumentResolver使用和原理
HandlerMethodArgumentResolver使用和原理
2020-11-07 17:19:00 【Lbj虞】
HandlerMethodArgumentResolver 用于将方法参数解析为参数值的策略接口,我们常说的自定义参数解析器,源码如下
/**
* Strategy interface for resolving method parameters into argument values in
* the context of a given request.
*
* @author Arjen Poutsma
* @since 3.1
* @see HandlerMethodReturnValueHandler
*/
public interface HandlerMethodArgumentResolver {
/**
* Whether the given {@linkplain MethodParameter method parameter} is
* supported by this resolver.
* @param parameter the method parameter to check
* @return {@code true} if this resolver supports the supplied parameter;
* {@code false} otherwise
*/
boolean supportsParameter(MethodParameter parameter);
/**
* Resolves a method parameter into an argument value from a given request.
* A {@link ModelAndViewContainer} provides access to the model for the
* request. A {@link WebDataBinderFactory} provides a way to create
* a {@link WebDataBinder} instance when needed for data binding and
* type conversion purposes.
* @param parameter the method parameter to resolve. This parameter must
* have previously been passed to {@link #supportsParameter} which must
* have returned {@code true}.
* @param mavContainer the ModelAndViewContainer for the current request
* @param webRequest the current request
* @param binderFactory a factory for creating {@link WebDataBinder} instances
* @return the resolved argument value, or {@code null} if not resolvable
* @throws Exception in case of errors with the preparation of argument values
*/
@Nullable
Object resolveArgument(MethodParameter parameter, @Nullable ModelAndViewContainer mavContainer,
NativeWebRequest webRequest, @Nullable WebDataBinderFactory binderFactory) throws Exception;
}
下面是我的一个小例子,主要是通过自定义参数解析器去将分页参数,提取出来封装成实体对象
1.首先自定义一个注解
/**
* @Author: 虞云波(18088704)
* @Date: 2020/11/7 10:18
* @Description:
*/
@Target(ElementType.PARAMETER)
@Inherited
@Documented
@Retention(RetentionPolicy.RUNTIME)
public @interface CustomArgumentPage {
}
2.写一个自定义参数解析器实现类
import com.suning.logistics.jwms.on.base.vo.PageBean;
import org.apache.commons.lang3.StringUtils;
import org.springframework.core.MethodParameter;
import org.springframework.web.bind.support.WebDataBinderFactory;
import org.springframework.web.context.request.NativeWebRequest;
import org.springframework.web.method.support.HandlerMethodArgumentResolver;
import org.springframework.web.method.support.ModelAndViewContainer;
import javax.servlet.http.HttpServletRequest;
/**
* @Author: 虞云波(18088704)
* @Date: 2020/11/7 10:14
* @Description:
*/
public class CustomArgumentResolvers implements HandlerMethodArgumentResolver {
@Override
public boolean supportsParameter(MethodParameter parameter) {
return parameter.hasParameterAnnotation(CustomArgumentPage.class);
}
@Override
public Object resolveArgument(MethodParameter parameter, ModelAndViewContainer mavContainer, NativeWebRequest webRequest, WebDataBinderFactory binderFactory) throws Exception {
HttpServletRequest request = webRequest.getNativeRequest(HttpServletRequest.class);
String page = request.getParameter("page");
String pageSize = request.getParameter("pageSize");
PageBean pageBean = new PageBean();
if (StringUtils.isNotEmpty(page)) {
pageBean.setCurrentPage(Integer.valueOf(page).intValue());
}
if (StringUtils.isNotEmpty(pageSize)) {
pageBean.setPageSize(Integer.valueOf(pageSize).intValue());
}
pageBean.setStartPage((pageBean.getCurrentPage()-1)*pageBean.getPageSize());
return pageBean;
}
}
3.将自定义解析器注入到IOC 中
@Bean
public CustomArgumentResolvers initCustomArgumentResolvers(){
return new CustomArgumentResolvers();
}
4.将自定义的参数解析器,注入到MvcConfigure中
@Configuration
public class WebMvcCustomConfigurer implements WebMvcConfigurer {
@Autowired
private CustomArgumentResolvers customArgumentResolvers;
/**
* Add resolvers to support custom controller method argument types.
* <p>This does not override the built-in support for resolving handler
* method arguments. To customize the built-in support for argument
* resolution, configure {@link RequestMappingHandlerAdapter} directly.
*
* @param resolvers initially an empty list
*/
@Override
public void addArgumentResolvers(List<HandlerMethodArgumentResolver> resolvers) {
resolvers.add(customArgumentResolvers);
}
}
5.编写测试Controller
@RequestMapping("/queryPageConfigList")
@ResponseBody
public Object getWarehouseCodeReadyonly(@CustomArgumentPage PageBean pageBean){
return warehouseConfigService.queryPage(pageBean,null);
}
6.效果如下
下面源码分析一波,讲述一下整个流程,
1.在Springboot AutoConfiguration 会自动注入 WebMvcAutoConfiguration
2.在WebMvcAutoConfiguration 中 静态内部类 EnableWebMvcConfiguration它继承了DelegatingWebMvcConfiguration这个类,这个很重要,中的 requestMappingHandlerAdapter方法,初始化HandlerAdapter
super.requestMappingHandlerAdapter()
3.在父类WebMvcConfigurationSupport 中,如下图 565行代码,会给HandlerAdapter 设置自定义的参数解析器
4.进入getArgumentResolvers方法,核心方法 addArgumentResolvers
5.是个空方法???,源码跟踪是进入了其子类 DelegatingWebMvcConfiguration 这个类中
6.继续代码跟踪下一步,会到WebMvcConfigurerComposite 中addArgumentResolvers 方法,会循环WebMvcConfigurer List 调用里面的 addArgumentResolvers 至于这个this.delegates 怎么初始化的,你们可以自己研究一下
7.最终调到我们自定义的WebMvcCustomConfigurer 中,将我们自定义的参数解析器,添加到整个参数解析List中
8.将数参数解析List 设置到HandlerAdapter 中
9.调用的时候,整个调用链如下图,刚兴趣的同学可以自己去分析一下
版权声明
本文为[Lbj虞]所创,转载请带上原文链接,感谢
https://my.oschina.net/u/3195939/blog/4707620
边栏推荐
- pc端与移动端适配解决方案之rem
- How to optimize the decoding performance of dynamsoft barcode reader
- 如何利用PopupWindow实现弹出菜单并解决焦点获取以及与软键盘冲突问题
- K-vim installation and the ycmd server shut down (restart with ': ycmrestartserver')
- And how to solve the conflict between pop-up menu and pop-up menu
- 深入浅出大前端框架Angular6实战教程(Angular6、node.js、keystonejs、
- Design pattern of facade and mediator
- Blazor 準備好為企業服務了嗎?
- Practice of Xiaoxiong school development board: real equipment access of smart street lamp sandbox experiment
- Python 3 operates the Jenkins module API
猜你喜欢
JS array the usage of array is all here (array method reconstruction, array traversal, array de duplication, array judgment and conversion)
Dynamsoft barcode reader v7.5!
如何利用PopupWindow实现弹出菜单并解决焦点获取以及与软键盘冲突问题
vue踩坑:axios使用this指针
聊聊先享後付
Stm32f030f4p6 compatible with smart micro mm32f031f4p6
gitlab 持续集成开发环境之CI/CD
你真的会使用搜索引擎吗?
如何创建交互式内核密度图表
OpenCV計算機視覺學習(10)——影象變換(傅立葉變換,高通濾波,低通濾波)
随机推荐
南京标志设计,logo设计公司
微服務 - 如何進行服務限流和熔斷
jenkins pipline stage 设置超时
LEADTOOLS如何检测,读取和写入条形码
Application of UHF RFID medical blood management system
心理咨询app开发所具备的优点与功能
使用RabbitMQ实现分布式事务
如何使用甘特图图层和筛选器
Dynamsoft barcode reader v7.5!
20个XR项目路演,近20个资本机构出席!诚邀您参加2020 Qualcomm XR生态合作伙伴大会
yum [Errno 256] No more mirrors to try 解决方法
Opencv computer vision learning (10) -- image transform (Fourier transform, high pass filter, low pass filter)
【原創】ARM平臺記憶體和cache對xenomai實時性的影響
9.集群之间服务通信 RoutingMesh
Two dimensional code location and alarm system of Expressway
PHP后门隐藏技巧
Blazor 準備好為企業服務了嗎?
Win7 how to quickly type CMD and get to the required directory
甘特图对活动进行分组教程
Nonvolatile MRAM memory used in all levels of cache