likes
comments
collection
share

一个好用的SpringBoot参数处理器-HandlerMethodArgumentResolver

作者站长头像
站长
· 阅读数 27

我本来想着处理分页参数不是用封装的方式就可以了吗,但是想着万一人家有更加隐式的方式呢,就点进去看了一下。

果然,里面介绍了一个我没有用过的接口- HandlerMethodArgumentResolver

这个类就很有意思,可以拦截接口参数,并修改入参值。

那我们就直接开始。

接口源码

源码非常简单,一看就知道怎么用的了:

public interface HandlerMethodArgumentResolver {
    boolean supportsParameter(MethodParameter parameter);

    @Nullable
    Object resolveArgument(MethodParameter parameter, @Nullable ModelAndViewContainer mavContainer, NativeWebRequest webRequest, @Nullable WebDataBinderFactory binderFactory) throws Exception;
}

首先supportsParameter就是判断接口参数是否需要进行处理,也就是执行resolveArgument方法。这里的入参MethodParameter就包含了调用的方法和处理的参数。

值得注意的是,一般情况下,supportsParameter对于一个入参都只会调用一次。当supportsParameter被调用后,HandlerMethodArgumentResolverComposite就会将其存放在argumentResolverCache中,用以作下次调用的缓存。

resolveArgument就是对这个入参进行处理,入参就是一堆请求信息,而出参就是对参数的设定值。

举例

我们知道@RequestParam可以设定默认值,那么对于@RequestBody该怎样设定默认值呢?可以这样:


public class MyHandlerMethodArgumentResolver implements HandlerMethodArgumentResolver {

    @Override
    public boolean supportsParameter(MethodParameter parameter) {
        // 只对被@RequestBody注解标记的参数进行默认值注入
        return parameter.hasParameterAnnotation(RequestBody.class);
    }

    @Override
    public Object resolveArgument(MethodParameter parameter, ModelAndViewContainer mavContainer, NativeWebRequest webRequest, WebDataBinderFactory binderFactory) throws Exception {
        Object value = getValue(parameter, webRequest);
        // TODO: modify value
        return value;
    }
    
    private Object getValue(MethodParameter parameter, NativeWebRequest request) {
        // TODO: 从request解析出parameter对应的值
        return null;
    }
}

使用

当然,我们只定义了一个参数处理器是不够的,SpringBoot并不会直接把这个实现类拿来用,我们还需要手动把实现类添加到处理器列表中:

@Configuration
public class WebConfig implements WebMvcConfigurer {
    @Override
    public void addArgumentResolvers(List<HandlerMethodArgumentResolver> resolvers) {
        resolvers.add(new MyHandlerMethodArgumentResolver());
    }
}

值得注意的是,如果你用的@RequestBody,你会发现参数解析并不会执行到我们自定义的方法里。这是因为RequestMappingHandlerAdapter适配器的控制。

这里不展开讲,只说明一点,我们自定义的处理器在默认27个处理器的倒数,也就是只有在其他处理器不处理的情况下才会执行我们的处理器。因此,我们可以采用这样的方式来提升我们处理器的优先级:

@Configuration
public class MockArgInitializer implements InitializingBean {

    @Autowired
    private RequestMappingHandlerAdapter requestMappingHandlerAdapter;
    @Autowired
    private MockArgResolver argResolver;

    @Override
    public void afterPropertiesSet() throws Exception {
        // 这里通过反射将MockArgResolver添加到第一位
        Field argumentResolvers = RequestMappingHandlerAdapter.class.getDeclaredField("argumentResolvers");
        HandlerMethodArgumentResolverComposite resolverComposite = (HandlerMethodArgumentResolverComposite) FieldUtil.getFieldValue(requestMappingHandlerAdapter, argumentResolvers);
        if (resolverComposite != null) {
            List<HandlerMethodArgumentResolver> resolvers = (List<HandlerMethodArgumentResolver>) FieldUtil.getFieldValue(resolverComposite, "argumentResolvers");
            if (resolvers != null) {
                resolvers.add(0, argResolver);
            }
        }
    }
}

不通过默认的WebMvcConfigurer,而是通过反射的方式手动控制顺序。至于为什么要用反射,你可以查看RequestMappingHandlerAdapter.getArgumentResolvers的源码就知道了。

其他

WebMvcConfigurer可以看出,里面其实还有很多的add方法,例如addReturnValueHandlers就是对应的更改方法返回值。

这里有一个基于HandlerMethodArgumentResolver的项目,可以参考下:mockapi-arg

转载自:https://juejin.cn/post/7368413596954755099
评论
请登录