SpringBoot 同一个接口同时支持 form 表单、form-data、json 的优雅写法[通俗易懂]

SpringBoot 同一个接口同时支持 form 表单、form-data、json 的优雅写法[通俗易懂]最近重写个项目遇到个比较棘手的问题,老项目是 PHP 接口,这个接口同时兼容 POST json 和 form 表单,更骚的是连 form-data 也兼容。。。因为写 PHP 请求的对接方代码不严谨

网上很多代码都是千篇一律的 cvs,相信我只要你认真看完我写的这篇,你就可以完全掌握这个知识点,这篇文章不适合直接 cvs,一定要先理解。

最近重写个项目遇到个比较棘手的问题,老项目是 PHP 接口,这个接口同时兼容 POST json 和 form 表单,更骚的是连 form-data 也兼容。。。因为写 PHP 请求的对接方代码不严谨。详见这里

而在 Java 中,一个接口只支持一种 content-type,json 就用 @RequestBody,form 表单就用 @RequestParam 或不写,form-data 就用 MultipartFile。

兼容版本

如果要把在一个接口中同时兼容三种,比较笨的办法就是获取 HttpServletRequest,然后自己再写方法解析。类似如下:

private Map<String, Object> getParams(HttpServletRequest request) {

    String contentType = request.getContentType();
    if (contentType.contains("application/json")) {
        // json 解析...
        return null;
    } else if (contentType.contains("application/x-www-form-urlencoded")) {
        // form 表单解析 ...
        return null;
    } else if (contentType.contains("multipart")) {
        // 文件流解析
        return null;
    } else {
         throw new BizException("不支持的content-type");
    } 

}

但是这样写有弊端

  1. 代码很丑,具体到解析代码又臭又长
  2. 只能返回固定 map 或者自己重新组装参数类
  3. 无法使用 @Valid 校验参数,像我这种几十个参数都要检验的简直是灾难

优雅版本

网上有 form 表单和 json 同时兼容的版本,但是没有兼容 form-data,我在这做一下补充。

1. 自定义注解

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

2. 自定义注解解析

public class GamePHPMethodProcessor implements HandlerMethodArgumentResolver {

    private GameFormMethodArgumentResolver formResolver;
    private GameJsonMethodArgumentResolver jsonResolver;

    public GamePHPMethodProcessor() {
        List<HttpMessageConverter<?>> messageConverters = new ArrayList<>();
        PHPMessageConverter PHPMessageConverter = new PHPMessageConverter();
        messageConverters.add(PHPMessageConverter);

        jsonResolver = new GameJsonMethodArgumentResolver(messageConverters);
        formResolver = new GameFormMethodArgumentResolver();
    }

    @Override
    public boolean supportsParameter(MethodParameter parameter) {
        GamePHP ann = parameter.getParameterAnnotation(GamePHP.class);
        return (ann != null);
    }

    @Override
    public Object resolveArgument(MethodParameter methodParameter, ModelAndViewContainer modelAndViewContainer, NativeWebRequest nativeWebRequest, WebDataBinderFactory webDataBinderFactory) throws Exception {
        ServletRequest servletRequest = nativeWebRequest.getNativeRequest(ServletRequest.class);
        String contentType = servletRequest.getContentType();
        if (contentType == null) {
            throw new IllegalArgumentException("不支持contentType");
        }

        if (contentType.contains("application/json")) {
            return jsonResolver.resolveArgument(methodParameter, modelAndViewContainer, nativeWebRequest, webDataBinderFactory);
        }

        if (contentType.contains("application/x-www-form-urlencoded")) {
            return formResolver.resolveArgument(methodParameter, modelAndViewContainer, nativeWebRequest, webDataBinderFactory);
        }

        if (contentType.contains("multipart")) {
            return formResolver.resolveArgument(methodParameter, modelAndViewContainer, nativeWebRequest, webDataBinderFactory);
        }

        throw new IllegalArgumentException("不支持contentType");
    }
}

3. 添加到 spring configuration


    @Bean
    public MyMvcConfigurer mvcConfigurer() {
        return new MyMvcConfigurer();
    }

    public static class MyMvcConfigurer implements WebMvcConfigurer {
        public void addArgumentResolvers(List<HandlerMethodArgumentResolver> resolvers) {
            resolvers.add(new GamePHPMethodProcessor());
        }
    }

4. form-data 的特殊处理

引入 jar 包

    <dependency>
      <groupId>commons-fileupload</groupId>
      <artifactId>commons-fileupload</artifactId>
      <version>1.3.1</version>
    </dependency>
    <dependency>
      <groupId>commons-io</groupId>
      <artifactId>commons-io</artifactId>
      <version>2.4</version>
    </dependency>

新增解析 bean

@Bean(name = "multipartResolver")
public MultipartResolver multipartResolver(){
    CommonsMultipartResolver resolver = new CommonsMultipartResolver();
    resolver.setDefaultEncoding("UTF-8");
    resolver.setResolveLazily(true);//resolveLazily属性启用是为了推迟文件解析,以在在UploadAction中捕获文件大小异常
    resolver.setMaxInMemorySize(40960);
    resolver.setMaxUploadSize(50*1024*1024);//上传文件大小 50M 50*1024*1024
    return resolver;
}

特殊说明,GameJsonMethodArgumentResolverGameFormMethodArgumentResolver 是我们自定义的 json 和 form 解析,如果你没有自定义的,使用 spring 默认的 ServletModelAttributeMethodProcessorRequestResponseBodyMethodProcessor 也可以。

只需将 @RequestParam 注解改为 @GamePHP,接口即可同时兼容三种 content-type。

其流程为,spring 启动的时候,MyMvcConfigurer 调用 addArgumentResolvers 方法将 GamePHPMethodProcessor 注入,接到请求时,supportsParameter 方法判断是否使用此 resolver,如果为 true,则进入 resolveArgument 方法执行。

至此我们可以得出一个结论,PHP 是世界上最垃圾的语言。写代码一时爽,维护火葬场。

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 举报,一经查实,本站将立刻删除。
转载请注明出处: https://daima100.com/12979.html

(0)

相关推荐

  • Python接口的重要性

    Python接口的重要性Python是一种高级语言,以其简洁的语法和灵活性而著称。在当前快速发展的互联网和移动应用生态体系中,Python的角色越来越关键。Python的主要优点之一是其完美的接口性,它提供了各种接口工具,其实现和开发都非常容易,而且非常有用,使得Python在实际应用场景中具有广泛的适用性和可扩展性。

    2024-08-15
    20
  • mysql密码错误-ERROR 1045 (28000): Access denied for user ‘root’@’localhost’ (using passwor:yes)

    mysql密码错误-ERROR 1045 (28000): Access denied for user ‘root’@’localhost’ (using passwor:yes) 一般这个错误是由密码错误引起,解决的办法自然就是重置密码。 假设我们使用的是root账户。 1.重置密码的第一步就是跳过MySQL的密码认证过程,方法如下: root 00:22:26~$ …

    2023-03-29
    146
  • 自学编程之前,你需要知道这些_自学程序员有没公司要

    自学编程之前,你需要知道这些_自学程序员有没公司要编程是报酬相当丰厚的行业,有不少的小伙伴对于编程是相当的感兴趣。**如果刚毕业还很年轻,对编程感兴趣,不妨参考w3cschool分享的编程自学方

    2022-12-14
    306
  • Python字典:高效存储和管理数据

    Python字典:高效存储和管理数据Python字典是一种可变容器模型,且内部元素无序,元素以键值对的形式存储。其中,键必须唯一,可以是任意不可变类型,例如整数、浮点数或字符串,值可以是任意类型的Python对象。字典可以通过键来快速访问其值,它比列表和元组等容器的访问效率更高。

    2023-12-14
    102
  • 跨区迁移需要什么手续_同步调用

    跨区迁移需要什么手续_同步调用近年来,云计算已成为主流,企业从自身利益出发,或是不愿意被单一云服务商锁定,或是业务和数据冗余,或是出于成本优化考虑,会尝试将部分或者全部业务从线下机房迁移到云或者从一个云平台迁移到另一个云平台,业务

    2023-05-12
    127
  • 四个福利性在线网站分享,每一个都让你欲罢不能……「建议收藏」

    四个福利性在线网站分享,每一个都让你欲罢不能……「建议收藏」随着社会的不断发展,我们网络上的一些资源就像雨后春笋一般的拔地而起,许多东西我们的来不及上车就被新的事物给掩盖了,但是对于一些好用的网站来说是真的很不错,小编今天就和你们分享一些最具福利性的在线网站!你们准备好了吗?老司机准备发车了:一.TinEye.com—图片搜索引擎如果你的电脑或者你的手机里有一张不知道哪里来的图片,你想要将图片的源地址给找出来该怎么做?一般的时候我们都是束手无策,但是有了这个网站就不一样,可以图片上传带上去进行查找…

    2023-08-21
    122
  • 荣耀FlyPods3体验 优秀的方式不仅一种「建议收藏」

    荣耀FlyPods3体验 优秀的方式不仅一种「建议收藏」  真无线蓝牙耳机以其功能的优越性和结构的便利性受到了大众的一致认可,笔者也没有忍住对真无线蓝牙耳机的心动。经过各方面比较入手了一款荣耀旗下的荣耀FlyPods3,经过一段时间的体验,发现这对耳机在…

    2023-03-05
    135
  • Redis API 介绍和使用(上)「终于解决」

    Redis API 介绍和使用(上)「终于解决」介绍5种数据结构之前,先了解一下Redis 的一些全局命令、数据结构和内部编码、单线程命令处理机制。 预备知识 1.全局命令 1.1 查看所有键 keys * 命令将所有的键输出。该命令会遍历所有键…

    2023-02-22
    115

发表回复

您的电子邮箱地址不会被公开。 必填项已用*标注