HandlerMethodArgumentResolver的简单使用

一、前言

现在项目有这样的一个需求,前端在请求头中传入userId,后端需要在controller中接收到此id的整个用户信息。

面对这样的需求,我们当然可以拿到userId后,在controller调用的service中请求数据库,从而拿到整个用户信息。不过这样重复的代码太多,丝毫没有灵活性可言。

因此我们考虑:能否在参数绑定的时候,就自动由userId往UserInfo对象中注入用户信息?当然是可以的,我们使用HandlerMethodArgumentResolver接口。


二、HandlerMethodArgumentResolver是什么?

HandlerMethodArgumentResolver,直接翻译是处理方法参数的解析器。说人话,这个类就是针对某一个方法列表的参数,比如UserInfo参数,原本前端只传入UserId,也就是UserInfo的其他参数都为null,但在解析器的改造下,UserInfo的整个属性都会有值,到时候我直接使用UserInfo参数。

下面看看HandlerMethodArgumentResolver的源码

public interface HandlerMethodArgumentResolver {
    boolean supportsParameter(MethodParameter var1);

    @Nullable
    Object resolveArgument(MethodParameter var1, @Nullable ModelAndViewContainer var2, NativeWebRequest var3, @Nullable WebDataBinderFactory var4) throws Exception;
}

由源码可知,HandlerMethodArgumentResolver是个接口,里面有两个抽象方法。

boolean supportsParameter(MethodParameter var1) 
是否支持此方法参数的解析,如果返回false,则不调用resolveArgument方法
Object resolveArgument(MethodParameter var1, @Nullable ModelAndViewContainer var2, NativeWebRequest var3, @Nullable WebDataBinderFactory var4) throws Exception

解析参数的方法,如果前端将userId放入请求头,则我们从NativeWebRequest中获取userId参数。该方法的返回值将自动注入到被注解标记的UserInfo中。


三、简单示例

我们的步骤分四步:类似于把大象放进冰箱

(1)自定义注解,被注解标记的类,我们才对它使用解析器。(我现在只要找大象这个动物)

@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.PARAMETER)
@Documented
public @interface UserId {
}

(2)自定义解析器,到底怎么去解析。(把大象放进冰箱的方式)

public class UserInfoResolver implements HandlerMethodArgumentResolver {
    @Override
    public boolean supportsParameter(MethodParameter methodParameter) {
        return methodParameter.hasParameterAnnotation(UserId.class);
    }

    @Override
    public Object resolveArgument(MethodParameter methodParameter, ModelAndViewContainer modelAndViewContainer, NativeWebRequest nativeWebRequest, WebDataBinderFactory webDataBinderFactory) throws Exception {
        HttpServletRequest request = nativeWebRequest.getNativeRequest(HttpServletRequest.class);
        Long userId = Long.valueOf(request.getHeader("userId"));
        //查询数据库或缓存
        UserInfo userInfo = new UserInfo();
        userInfo.setUserId(userId);
        userInfo.setUserName("tom");
        userInfo.setUserAge(20);
        return userInfo;
    }
}

(3)注册解析器(给冰箱通电)

@Configuration
public class ConfigAdapter extends WebMvcConfigurerAdapter {

    @Override
    public void addArgumentResolvers(List<HandlerMethodArgumentResolver> argumentResolvers) {
        super.addArgumentResolvers(argumentResolvers);
        argumentResolvers.add(new UserInfoResolver());
        //当然可以增加一连串的解析器
        argumentResolvers.add(new FoodInfoResolver());
    }
}

(4)验证解析器(打开冰箱门看一看,大象还在里面吗)

@RestController
@RequestMapping("/user")
public class UserController {

    @RequestMapping("/info")
    public void getUserInfo(@UserId UserInfo userInfo, @FoodId FoodInfo foodInfo) {
        System.out.println(userInfo.toString());
        System.out.println(foodInfo.toString());
    }
}

postman请求:

控制台输出:

大功告成!