最近在尝试用PUT方法的请求时一直产生400,415,500错误,弄了半天(真的是半天),尝试了各种办法,现在终于解决了,为了防止忘记,在此记录下

下面是一步步解决的步骤。(还有许多我略过了)如果只想看结果滑到底即可。

一开始遇到的是400错误,前端是这样写的

  $.ajax({
type: "POST", // 这里必须写POST
url: "/team/" + teamId,
data: $("#myForm").serialize() + "&_method=PUT",
dataType: "json",
success: function (vo) {
if (vo.code === 200) {
window.location.href = "/pages/team/list.html?pageNum=" + pageNum + "&pageSize=" + pageSize;
} else {
alert("更新失败!" + vo.msg);
}
}
});

我先在后台 TeamController 中对应的方法上打断点

  @ResponseBody
@RequestMapping(value = "team/{id}", method = RequestMethod.PUT)
public ResultVO<?> update(@PathVariable Integer id, Team team) {
team.setTeamId(id); // --------------------- 在这一行打断点
if (id != null && teamService.update(team)) {
return new ResultVO<>();
}
return new ResultVO<>("球队ID 不存在!", 500);
}

发现根本没有走到这里。

于是我就检查是否是 HiddenHttpMethodFilter 没有配置,然而 web.xml 写的好好的

<filter>
<filter-name>httpMethodFilter</filter-name>
<filter-class>org.springframework.web.filter.HiddenHttpMethodFilter</filter-class>
</filter>
<filter-mapping>
<filter-name>httpMethodFilter</filter-name>
<url-pattern>/*</url-pattern>
</filter-mapping>

然后我按Ctrl点进 HiddenHttpMethodFilter 里,在里面打断点

  public static final String DEFAULT_METHOD_PARAM = "_method";
private String methodParam = DEFAULT_METHOD_PARAM; @Override
protected void doFilterInternal(HttpServletRequest request, HttpServletResponse response, FilterChain filterChain)
throws ServletException, IOException { HttpServletRequest requestToUse = request; // --------------------- 在这一行打断点 if ("POST".equals(request.getMethod()) && request.getAttribute(WebUtils.ERROR_EXCEPTION_ATTRIBUTE) == null) {
String paramValue = request.getParameter(this.methodParam); // ---- (1) ----- //
if (StringUtils.hasLength(paramValue)) {
String method = paramValue.toUpperCase(Locale.ENGLISH);
if (ALLOWED_METHODS.contains(method)) {
requestToUse = new HttpMethodRequestWrapper(request, method);
}
}
} filterChain.doFilter(requestToUse, response);
}

发现它有走到这里, 但是走到 (1)这一行的时候, paramValue 没有取到值,也就是说 request 没有 _method 的值,可是我前端有写啊!折磨了好久,把 target 文件夹删了重开,paramValue 竟然就能取到值了!?离谱。

但是问题没有解决,依然还是400错误。

我又尝试把 _method=PUT 删掉,用 POST 请求,这样就可以正常访问到。但是这不是我要的结果,POST 请求已经被其他方法占用了。

一番百度,好几人说添加一行 contentType:"application/json" ,于是我就添加一行

  $.ajax({
type: "POST", // 这里必须写POST
url: "/team/" + teamId,
data: $("#myForm").serialize() + "&_method=PUT",
dataType: "json",
contentType: "multipart/form-data", // 添加了这一行
success: function (vo) {
if (vo.code === 200) {
window.location.href = "/pages/team/list.html?pageNum=" + pageNum + "&pageSize=" + pageSize;
} else {
alert("更新失败!" + vo.msg);
}
}
});

这回不是400了,变成了415错误。害,继续搞呗。

继续百度,有人说后台把 Team 对象改成一个map集合,像这样 : @RequestBody Map<String, String> map ,不出意料,改了之后没有任何作用。

我尝试把 type 由 “POST” 改成 “PUT” ,像这样


  $.ajax({
type: "PUT",
url: "/team/" + teamId,
data: $("#myForm").serialize(),
dataType: "json",
// contentType: "multipart/form-data",
success: function (vo) {
if (vo.code === 200) {
window.location.href = "/pages/team/list.html?pageNum=" + pageNum + "&pageSize=" + pageSize;
} else {
alert("更新失败!" + vo.msg);
}
}
});

这样后台只取到了 id, Team 里的信息没有取到。

我又怀疑是版本的问题。我将 target 清除,改成使用 maven 的 tomcat 插件启动(之前是用 tomcat 9.0.52),插件是 7 的版本。然而,很不幸,问题还是没有解决(我要吐槽一下这个tomcat插件,连页面都进不去,用下载的tomcat 9 就没问题)。

后来看到一篇说 JSON 的属性也要加上双引号 像这样 { “aaa”: “bbb” },而不是 { aaa: "bbb" },我就想,把 data 换成 JSON 格式试试,竟然就真的可以了!太奇怪了吧。

  $.ajax({
type: "POST", // 这里必须写POST
url: "/team/" + teamId,
data: {
"teamId": $teamId.val(),
"teamName": $teamName.val(),
"chineseName": $chineseName.val(),
"coach": $coach.val(),
"stadium": $stadium.val(),
"location": $location.val(),
"createTime": $createTime.val(),
"area": $area.val(),
"_method": "PUT" // 不要忘记写这一行,将来能把 POST 改成 PUT
},
dataType: "json",
// contentType: "multipart/form-data",
success: function (vo) {
if (vo.code === 200) {
window.location.href = "/pages/team/list.html?pageNum=" + pageNum + "&pageSize=" + pageSize;
} else {
alert("更新失败!" + vo.msg);
}
}
});

然而事情还没完,我把前端改成原先的模样,他竟然又可以了!看来不是前端的问题。那究竟是哪里的问题呢?首先排除是target的问题,因为前面删掉重新生成过了。应该是我哪里改对了又忘了。。。

后来,第二天,我又在另一个模块遇到了400错误。这回我把 ajax 的 data 改成 JSON 格式也无济于事了,之前的各种办法试了也没用,就卡死在这里了。

又过了一天,我想再看看是什么原因。既然一个可以用,另一个不可以,那对比一下两者之间有什么不同不就行了吗。首先前端的 ajax 都是一样的,检查 Controller,注解也都一样,没有缺失什么。

我又把目光转向了 pojo 类:

 // 这个是可以用的
@DateTimeFormat(pattern = "yyyy-MM-dd")
@JsonFormat(pattern = "yyyy-MM-dd")
private Date createTime; // 这个会报 400 错误
@JsonFormat(pattern = "yyyy-MM-dd")
private Date birthday;

两个 pojo 类都有 Date 类型的属性,但是其中一个少了@DateTimeFormat(pattern = "yyyy-MM-dd")注解。马上安排,我加上注解之后,果然可以正常运行了。

由此可见,@DateTimeFormat(pattern = "yyyy-MM-dd")注解是将浏览器发送到后台的数据进行打包的,如果没有的话在接受数据时就会发生问题。

而@JsonFormat(pattern = "yyyy-MM-dd")是将后台要发送给前端的数据进行打包的,如果没有这个注解前端显示就会有问题。

SSM PUT请求导致的400,415,500问题的更多相关文章

  1. 网络请求返回HTTP状态码(404,400,500)

    HTTP状态码(HTTP Status Code) 一些常见的状态码为: - 服务器成功返回网页 - 请求的网页不存在 - 服务不可用 所有状态解释: 1xx(临时响应) 表示临时响应并需要请求者继续 ...

  2. 解决ajax请求(SpringMVC后台)响应415/400/405错误

    解决ajax请求(SpringMVC后台)响应415/400/405错误 后端代码 bean public class user { private String username; private ...

  3. HTTP状态码100、200、300、400、500、600的含义

    1xx (临时响应)表示临时响应并需要请求者继续执行操作的状态代码. 100 (继续) 请求者应当继续提出请求. 服务器返回此代码表示已收到请求的第一部分,正在等待其余部分. 101 (切换协议) 请 ...

  4. HTTP状态码,400,404,500,503

    HTTP状态码(HTTP Status Code) 一些常见的状态码为: 200 - 服务器成功返回网页 400 服务器不理解请求的语法 404 - 请求的网页不存在 503 - 服务不可用 所有状态 ...

  5. HTTP常见状态码(404、400、500)等错误

    一些常见的状态码为: 200 - 服务器成功返回网页 404 - 请求的网页不存在 503 - 服务不可用 详细分解: 1xx(临时响应) 表示临时响应并需要请求者继续执行操作的状态代码. 代码 说明 ...

  6. 关于web资金系统提现安全保护,防止极快的重复并发请求导致重复提现的解决思路

    关于WEB金融系统中的提现安全问题很多人没有深入思想,导致有漏洞,常常会遇到有些人遇到被攻击到导资金损失的麻烦,     其实要彻底解决重复并发请求 导致重复提现问题,是需要花点心思的,并没有看起来的 ...

  7. Asp.net并发请求导致的数据重复插入问题

    前段时间工作中,有客户反应了系统中某类待办重复出现两次的情况.我核实了数据之后,分析认为是并发请求下导致的数据不一致性问题,并做了重现.其实这并不是一个需要频繁调用的功能,但是客户连续点击了两次,导致 ...

  8. java异常处理及400,404,500错误处理

        java代码中经常碰到各种需要处理异常的时候,比如什么IOException  SQLException  NullPointException等等,在开发web项目中,遇到异常,我现在做的就 ...

  9. http statusCode(状态码) 200、300、400、500序列

    201-206都表示服务器成功处理了请求的状态代码,说明网页可以正常访问.        200(成功)  服务器已成功处理了请求.通常,这表示服务器提供了请求的网页.        201(已创建) ...

  10. http statusCode(状态码) 200、300、400、500序列详解

    201-206都表示服务器成功处理了请求的状态代码,说明网页可以正常访问.200(成功) 服务器已成功处理了请求.通常,这表示服务器提供了请求的网页.201(已创建) 请求成功且服务器已创建了新的资源 ...

随机推荐

  1. gitee部署

    1.安装git 下载地址:https://npm.taobao.org/mirrors/git-for-windows/,拉到最下方选最新版,点击进去后选择对应windows版本的exe文件,默认安装 ...

  2. centos7.6安装rz命令上传文件

    centos7.6安装rz命令 1.执行安装命令:yum -y install lrzsz 2.进行上传操作:rz   跳转到本地文件选择后即可上传

  3. CSP-S T3函数调用

    函数是各种编程语言中一项重要的概念,借助函数,我们总可以将复杂的任务分解成一个个相对简单的子任务,直到细化为十分简单的基础操作,从而使代码的组织更加严密.更加有条理.然而,过多的函数调用也会导致额外的 ...

  4. 【git】git子模块操作-添加子模块与克隆子模块

    https://git-scm.com/book/zh/v2/Git-%E5%B7%A5%E5%85%B7-%E5%AD%90%E6%A8%A1%E5%9D%97 git submodule upda ...

  5. kmp失效函数(测试博客的编辑器)

    1 int *find_next(char *p) 2 { 3 int k = -1; //right shift i-k 4 int i = 0; 5 int m = strlen(p); 6 in ...

  6. js右键生成菜单

    <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8&quo ...

  7. Beginning IOS 7 Development Exploring the IOS SDK - Navigation Controllers and Table Views

    Note You may notice that the familyNames property is declared using the copy keyword instead of stro ...

  8. Solution Set - 杭电多校 2022 Day2 一句话题解

    A:看了题就很容易想到虚树吧,建出虚树后考虑整体扫一遍虚树,注意到这是一棵根向树,那么统计其实十分简单,将对 \(C\) 类节点的标记下放,\(A,B\) 类节点同时上传,如果在 DFS 的过程中发现 ...

  9. Es6中模块引入的相关内容

    注意:AMD规范和commonJS规范 1.相同点:都是为了模块化. 2.不同点:AMD规范则是非同步加载模块,允许指定回调函数.CommonJS规范加载模块是同步的,也就是说,只有加载完成,才能执行 ...

  10. unix:///var/run/supervisor.sock no such file报错解决办法

    报错 unix:///var/run/supervisor.sock no such file 原因 /var/run/supervisor.sock已被清理. 解决办法 关闭supervisor:p ...