Java自学者论坛

 找回密码
 立即注册

手机号码,快捷登录

恭喜Java自学者论坛(https://www.javazxz.com)已经为数万Java学习者服务超过8年了!积累会员资料超过10000G+
成为本站VIP会员,下载本站10000G+会员资源,会员资料板块,购买链接:点击进入购买VIP会员

JAVA高级面试进阶训练营视频教程

Java架构师系统进阶VIP课程

分布式高可用全栈开发微服务教程Go语言视频零基础入门到精通Java架构师3期(课件+源码)
Java开发全终端实战租房项目视频教程SpringBoot2.X入门到高级使用教程大数据培训第六期全套视频教程深度学习(CNN RNN GAN)算法原理Java亿级流量电商系统视频教程
互联网架构师视频教程年薪50万Spark2.0从入门到精通年薪50万!人工智能学习路线教程年薪50万大数据入门到精通学习路线年薪50万机器学习入门到精通教程
仿小米商城类app和小程序视频教程深度学习数据分析基础到实战最新黑马javaEE2.1就业课程从 0到JVM实战高手教程MySQL入门到精通教程
查看: 417|回复: 0

SpringMVC 完美解决PUT请求参数绑定问题(普通表单和文件表单)

[复制链接]
  • TA的每日心情
    奋斗
    2024-11-24 15:47
  • 签到天数: 804 天

    [LV.10]以坛为家III

    2053

    主题

    2111

    帖子

    72万

    积分

    管理员

    Rank: 9Rank: 9Rank: 9

    积分
    726782
    发表于 2021-4-28 10:40:57 | 显示全部楼层 |阅读模式

    一 解决方案

    修改web.xml配置文件 将下面配置拷贝进去(在原有的web-app节点里面配置 其它配置不变)

    <!-- 处理PUT提交参数(只对基础表单生效) -->
    <filter>
        <filter-name>httpPutFormContentFilter</filter-name>
        <filter-class>org.springframework.web.filter.HttpPutFormContentFilter</filter-class>
    </filter>
    <filter-mapping>
        <filter-name>httpPutFormContentFilter</filter-name>
        <!-- 拦截所有 -->
        <url-pattern>/*</url-pattern>
    </filter-mapping>

    写一个PostAndPutCommonsMultipartResolver继承CommonsMultipartResolver 重写isMultipart()

    /**
     * 处理PUT提交参数(只对文件表单生效)
     * Created by Hy on 2018/9/30.
     */
    public class PostAndPutCommonsMultipartResolver extends CommonsMultipartResolver {
    
        @Override
        public boolean isMultipart(HttpServletRequest request) {
            if ("POST".equalsIgnoreCase(request.getMethod()) || "PUT".equalsIgnoreCase(request.getMethod())) {
                return FileUploadBase.isMultipartContent(new ServletRequestContext(request));
            }
            return false;
        }
    
    }

    修改spring-mvc.xml配置文件 将下面配置拷贝进去(在原有的beans节点里面配置 其它配置不变)

    <!-- 配置文件上传实现类 -->
    <bean id="multipartResolver" class="com.hy.mm.manager.resolver.PostAndPutCommonsMultipartResolver">
        <!-- 设定默认编码 -->
        <property name="defaultEncoding" value="UTF-8" />
        <!-- 文件上传大小(单位B) 30M = 30 * 1024 * 1024 -->
        <property name="maxUploadSize" value="31457280" />
    </bean>

    写一个Controller

    /**
     * PUT请求
     * Created by Hy on 2018/9/30.
     */
    @Controller
    public class PutController {
    
        @PutMapping("/put/normal") @ResponseBody
        public String normalForm(String name, Integer age) {
            System.out.println("name = " + name);
            System.out.println("age = " + age);
            return "ok";
        }
    
        @PutMapping("/put/file") @ResponseBody
        public String fileForm(String name, MultipartFile file) throws Exception {
            System.out.println("name = " + name);
            if (null != file && !file.isEmpty()) {
                System.out.println("file = " + file.getSize());
                // 保存图片
                String fileName = UUID.randomUUID().toString().replace("-", ""); //文件名
                String extension = FilenameUtils.getExtension(file.getOriginalFilename()); //扩展名 不包含(.)
                file.transferTo(new File("/Users/HUANGYI/Downloads/" + fileName + "." + extension));
                return "ok";
            }
            return "error";
        }
    
    }

    以上就能完美解决PUT请求参数绑定问题 赶时间的老哥可以忽略下文

     

    二 解决思路

    先bb一下起因

    我最近再重构一个自己的项目 打算把接口交互修改成RESTful风格 浅显的说一下RESTful风格 增删改查对应POST DELETE PUT GET请求

    环境

    客户端: Android 使用Retrofit发起请求

    服务端: Java 使用SpringMVC处理请求

    思路

    客户端使用PUT请求发送表单数据 不管是普通表单还是文件表单 服务端Controller层参数绑定均为null

    但是 客户端使用PUT请求发送非文件数据携带在Url上(类似GET请求) 服务端Controller层参数就能接收到

    为了避免重复造轮子 我用Google解决了普通表单数据接收不到 也就是使用上面说的org.springframework.web.filter.HttpPutFormContentFilter就可以解决该问题

    但是 文件表单数据还是接收不到 Google也不好用了 不知道是不是我姿势不对

    自己尝试解决吧

    先验证文件表单数据到底写入请求体没有?

    我用logging-interceptor和Charles观察了好几遍请求 确认了数据确实已经写入了请求体

    那么 问题肯定就出现在SpringMVC的文件参数绑定上

    仔细观察org.springframework.web.multipart.commons.CommonsMultipartResolver

    其中 isMultipart()是一个比较重要的方法 用来判断请求是否包含多部分内容 也就是判断是否是文件表单 深入观察一下该方法的实现

    真相大白 该方法默认POST请求才可能包含多部分内容

    使用上面说的PostAndPutCommonsMultipartResolver就可以解决该问题

    Android客户端核心代码

    /**
     * ...
     * Created by Hy on 2018/9/30.
     */
    public interface PutApi {
    
        @PUT("/put/normal") @FormUrlEncoded
        Call<ResponseBody> normal(@Field("name") String name, @Field("age") Integer age);
    
        @PUT("/put/file") @Multipart
        Call<ResponseBody> file(@Part("name") String name, @Part MultipartBody.Part file);
    }
    @Override
    public void onClick(View view) {
        switch (view.getId()) {
            case R.id.normal:
                Call<ResponseBody> normalCall = mApi.normal("黄祎", 18);
                normalCall.enqueue(new Callback<ResponseBody>() {
                    @Override
                    public void onResponse(Call<ResponseBody> call, Response<ResponseBody> response) {
                        try {
                            Log.i("HUANG", "code = " + response.code());
                            if (null != response.body())
                                Log.i("HUANG", "body = " + response.body().string());
    
                        } catch (IOException e) {
                            e.printStackTrace();
                        }
                    }
    
                    @Override
                    public void onFailure(Call<ResponseBody> call, Throwable t) {
                        Log.i("HUANG", "t = " + t.getMessage());
                    }
                });
                break;
    
            case R.id.file:
                RequestBody fileBody = RequestBody.create(MediaType.parse("application/octet-stream"), copy());
                MultipartBody.Part body = MultipartBody.Part.createFormData("file", "a.mp4", fileBody);
                Call<ResponseBody> fileCall = mApi.file("黄祎", body);
                fileCall.enqueue(new Callback<ResponseBody>() {
                    @Override
                    public void onResponse(Call<ResponseBody> call, Response<ResponseBody> response) {
                        try {
                            Log.i("HUANG", "code = " + response.code());
                            if (null != response.body())
                                Log.i("HUANG", "body = " + response.body().string());
    
                        } catch (IOException e) {
                            e.printStackTrace();
                        }
                    }
    
                    @Override
                    public void onFailure(Call<ResponseBody> call, Throwable t) {
                        Log.i("HUANG", "t = " + t.getMessage());
                    }
                });
                break;
        }
    }

    总结

    虽然只是寥寥几句 但是我走完这几步也花了一下午时间 哈哈哈 技术有限技术有限

    希望能帮助到你 如果你的问题得到解决 请给个推荐点个赞 这样能帮助到更多人 毕竟搜索不到解决方案的时候太痛苦了

     

    哎...今天够累的,签到来了1...
    回复

    使用道具 举报

    您需要登录后才可以回帖 登录 | 立即注册

    本版积分规则

    QQ|手机版|小黑屋|Java自学者论坛 ( 声明:本站文章及资料整理自互联网,用于Java自学者交流学习使用,对资料版权不负任何法律责任,若有侵权请及时联系客服屏蔽删除 )

    GMT+8, 2025-2-4 14:06 , Processed in 0.058447 second(s), 28 queries .

    Powered by Discuz! X3.4

    Copyright © 2001-2021, Tencent Cloud.

    快速回复 返回顶部 返回列表