Spring MVC策略模式之MethodArgumentResolver源码解析

 更新时间:2023年03月30日 09:00:37   作者:这堆干货有点猛  
这篇文章主要为大家介绍了Spring MVC策略模式之MethodArgumentResolver源码解析,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪

正文  

Spring MVC 是一个基于 MVC 设计模式的Web框架,它的核心就是 DispatcherServlet,它相当于请求的中央处理器。在 DispatcherServlet 中,它使用了 MethodArgumentResolver 来解析方法参数。

       MethodArgumentResolver 采用一种策略模式,在 Handler 的方法被调用前,Spring MVC 会自动将 HTTP 请求中的参数转换成方法参数。MethodArgumentResolver 接口的作用就是允许开发人员自定义参数解析器,以便更好地解析 HTTP 请求中的参数。

例子

       我们可以通过实现 MethodArgumentResolver 接口来创建自己的参数解析器。MethodArgumentResolver 通过 supportsParameter() 这个方法用来判断参数是否可以被当前解析器解析。如果返回 true,则调用 resolveArgument() 方法来解析参数。

下面是一个简单的例子,演示如何实现一个自定义的 MethodArgumentResolver:

public class CustomArgumentResolver implements MethodArgumentResolver {
    @Override
    public boolean supportsParameter(MethodParameter parameter) {
        return parameter.getParameterType().equals(CustomObject.class);
    }
    @Override
    public Object resolveArgument(MethodParameter parameter, ModelAndViewContainer mavContainer, NativeWebRequest webRequest, WebDataBinderFactory binderFactory) throws Exception {
        CustomObject customObject = new CustomObject();
        customObject.setName(webRequest.getParameter("name"));
        customObject.setAge(Integer.parseInt(webRequest.getParameter("age")));
        return customObject;
    }
}

       在上面的例子中,我们实现了一个 CustomArgumentResolver,用来解析 CustomObject 类型的参数。supportsParameter() 方法判断参数类型是否为 CustomObject 类型,如果是则返回true。resolveArgument() 方法用来解析参数,将请求中的 "age" 和 "name" 参数设置到 CustomObject 对象中。

源码分析

       Spring MVC 中有很多默认的参数解析器,比如 RequestParamMethodArgumentResolver、PathVariableMethodArgumentResolver、ModelMethodProcessor 等。下面我们来看一下这些解析器的源码实现。

RequestParamMethodArgumentResolver:

public class RequestParamMethodArgumentResolver implements HandlerMethodArgumentResolver {
    @Override
    public boolean supportsParameter(MethodParameter parameter) {
        return parameter.hasParameterAnnotation(RequestParam.class);
    }
    @Override
    public Object resolveArgument(MethodParameter parameter, ModelAndViewContainer mavContainer, NativeWebRequest webRequest, WebDataBinderFactory binderFactory) throws Exception {
        // 获取注解
        RequestParam annotation = parameter.getParameterAnnotation(RequestParam.class);
        String paramName = annotation.value();
        String defaultValue = annotation.defaultValue();
        boolean required = annotation.required();
        String[] paramValues = webRequest.getParameterValues(paramName);
        if (paramValues == null || paramValues.length == 0) {
            if (required) {
                throw new MissingServletRequestParameterException(paramName, parameter.getParameterType().getSimpleName());
            }
            return defaultValue;
        }
        if (paramValues.length == 1) {
            return convertIfNecessary(paramValues[0], parameter.getParameterType());
        }
        return Arrays.stream(paramValues).map(value -> convertIfNecessary(value, parameter.getParameterType())).collect(Collectors.toList());
    }
}

       RequestParamMethodArgumentResolver 用来解析请求中的 @RequestParam 注解参数

supportsParameter() 方法判断参数是否有 @RequestParam 注解,如果有则返回 true。

resolveArgument()方法解析参数,获取 RequestParam 注解的 value、defaultValue 和 required 属性,然后根据参数名从请求中获取参数值,最后将参数值转换成目标类型。

PathVariableMethodArgumentResolver:

public class PathVariableMethodArgumentResolver implements HandlerMethodArgumentResolver {
    @Override
    public boolean supportsParameter(MethodParameter parameter) {
        return parameter.hasParameterAnnotation(PathVariable.class);
    }
    @Override
    public Object resolveArgument(MethodParameter parameter, ModelAndViewContainer mavContainer, NativeWebRequest webRequest, WebDataBinderFactory binderFactory) throws Exception {
        PathVariable annotation = parameter.getParameterAnnotation(PathVariable.class);
        String attributeName = annotation.value();
        Map<String, String> uriTemplateVariables = (Map<String, String>) webRequest.getAttribute(HandlerMapping.URI_TEMPLATE_VARIABLES_ATTRIBUTE, RequestAttributes.SCOPE_REQUEST);
        String attributeValue = uriTemplateVariables.get(attributeName);
        return convertIfNecessary(attributeValue, parameter.getParameterType());
    }
}

       PathVariableMethodArgumentResolver 用来解析请求中的 @PathVariable 注解参数

supportsParameter() 方法判断参数是否有 @PathVariable 注解,如果有则返回 true。

resolveArgument() 方法解析参数,获取 @PathVariable 注解的 value 属性,然后从请求中获取 URI 模板变量的值,最后将变量值转换成目标类型。

ModelMethodProcessor:

public class ModelMethodProcessor implements HandlerMethodReturnValueHandler {
    @Override
    public boolean supportsReturnType(MethodParameter returnType) {
        return Model.class.isAssignableFrom(returnType.getParameterType());
    }
    @Override
    public void handleReturnValue(Object returnValue, MethodParameter returnType, ModelAndViewContainer mavContainer, NativeWebRequest webRequest) throws Exception {
        Model model = mavContainer.getModel();
        model.addAllAttributes((Map<String, ?>) returnValue);
    }
}

用来处理 Handler 方法的返回值,并将返回值添加到 Model中。

supportsReturnType() 方法判断返回类型是否为 Model 类型或其子类,如果是则返回 true。

handleReturnValue() 方法将返回值转换成 Map 类型,然后将 Map 中的键值对添加到 Model 中。

总结

       MethodArgumentResolver 的主要作用就是将请求参数转换为 Handler 方法的参数。在Spring MVC中,有很多默认的MethodArgumentResolver实现,例如RequestParamMethodArgumentResolver、PathVariableMethodArgumentResolver、ModelAttributeMethodArgumentResolver 等,这些默认的 MethodArgumentResolver 实现可以满足大多数场景的需求,如果需要自定义 MethodArgumentResolver 实现,可以通过实现 MethodArgumentResolver 接口来实现,这些都遵循了策略模式的设计原则,具有较好的可扩展性和可维护性

以上就是Spring MVC策略模式之MethodArgumentResolver源码解析的详细内容,更多关于Spring MVC MethodArgumentResolver的资料请关注脚本之家其它相关文章!

相关文章

  • 如何使用java修改文件所有者及其权限

    如何使用java修改文件所有者及其权限

    这篇文章主要介绍了如何使用java修改文件所有者及其权限,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下
    2019-12-12
  • IDEA快速搭建Java开发环境的教程图解

    IDEA快速搭建Java开发环境的教程图解

    这篇文章主要介绍了IDEA如何快速搭建Java开发环境,本文通过图文并茂的形式给大家介绍的非常详细,具有一定的参考借鉴价值,需要的朋友可以参考下
    2019-11-11
  • 深入分析Java异常

    深入分析Java异常

    本篇文章给大家详细分享了关于Java异常的相关知识点,对此有需要的朋友跟着学习下吧。
    2018-05-05
  • JAVA使用动态代理对象进行敏感字过滤代码实例

    JAVA使用动态代理对象进行敏感字过滤代码实例

    这篇文章主要介绍了JAVA使用动态代理对象进行敏感字过滤代码实例,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下
    2019-09-09
  • SpringBoot整合PostgreSQL的示例代码

    SpringBoot整合PostgreSQL的示例代码

    本文主要介绍了SpringBoot整合PostgreSQL的示例代码,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2023-07-07
  • Java动态代理和反射机制详解

    Java动态代理和反射机制详解

    这篇文章主要介绍了Java动态代理和反射机制,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2019-03-03
  • JAVA JVM面试题总结

    JAVA JVM面试题总结

    JVM 可以屏蔽与具体操作系统平台相关的信息,使 Java 程序只需生成在 Java 虚拟机上运行的目标代码,就可以在不同的平台上运行。这篇文章主要介绍了JAVA JVM面试题总结,大家可以参考一下
    2021-08-08
  • Java执行cmd命令的举例与注意事项

    Java执行cmd命令的举例与注意事项

    Java应用程序主要是通过Runtime和Process两个类来执行cmd命令,下面这篇文章主要给大家介绍了关于Java执行cmd命令的方法与注意事项,文中通过实例代码介绍的非常详细,需要的朋友可以参考下
    2022-02-02
  • 如何使用JCTools实现Java并发程序

    如何使用JCTools实现Java并发程序

    这篇文章主要介绍了如何使用JCTools实现Java并发程序,帮助大家更好的理解和学习使用Java,感兴趣的朋友可以了解下
    2021-03-03
  • Spring+Http请求+HttpClient实现传参

    Spring+Http请求+HttpClient实现传参

    这篇文章主要介绍了Spring+Http请求+HttpClient实现传参,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2020-03-03

最新评论