我有一个在 Tomcat 9 下运行的 Java servlet,作为正常流程的一部分,它调用 HttpServletResponse#setStatus()
几次。
在 Tomcat 9.0.0.M26 和 Java 8(1.8.0u144,Tomcat 报告为 1.8.0_144-b01
)上运行时,这工作正常。
在带有 Java 10.0.1 的 Tomcat 9.0.8.0 上运行时(Tomcat 报告为 10.0.1+10
),似乎在响应对象上调用 setStatus() 实际上只会导致设置响应状态一次,之后 HTTP 状态就不能再改变了。然而,通过 HttpServletResponse#setHeader() 发送到客户端的其他 header 似乎不受此影响;即使在 setStatus() 不再执行任何操作之后,setHeader() 仍能成功添加 header 。没有发送可能导致 HTTP header 终止的中间输出数据。
这是一个最小的工作示例:
package org.example;
import java.io.IOException;
import javax.servlet.ServletException;
import javax.servlet.annotation.WebServlet;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
@WebServlet("/HttpResponseStatusTestServlet")
public class HttpResponseStatusTestServlet extends HttpServlet {
protected void doGet(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException {
response.getWriter().append("Testing ");
response.setStatus(505);
response.setStatus(506);
response.getWriter()
.append("Served at: ")
.append(request.getContextPath())
.append(" with: ")
.append(Integer.toString(response.getStatus()));
}
}
调用此 servlet 时,我希望返回字符串 Testing Served at: ... with: 506
因为最后一个 HTTP 状态设置在 getStatus()
之前电话是506
.返回给客户端的 HTTP 状态代码同样应该是 506
.
但是,我最终得到的是 Testing Served at: ... with: 505
和一个 505
HTTP 状态。就好像第二个 setStatus() 调用根本不存在。
无论是否使用Testing
,结果都是一样的(除了输出开头存在response.getWriter().append("Testing ");
)在包含 setStatus() 调用之前,因此它似乎与提前终止 HTTP 响应 header 无关。
没有任何迹象表明我可以看到第二次 setStatus() 调用以任何方式失败,甚至它曾经在那里;似乎在第一次调用 setStatus() 之后的任何地方,在响应对象上调用 setStatus() 完全没有任何作用。
response.isCommitted()
的返回值是false
在有问题的服务器上贯穿上述 servlet:在 getWriter().append("Testing ");
之后在 setStatus(505)
之后调用打电话,在setStatus(506)
之后打电话。
我意识到为同一个请求多次调用 setStatus() 可能有点不正统,但是:
- 这真的不应该与 Tomcat 9.0.8 和 Java 10 一起工作吗,因为它可以与 Tomcat 9.0.0 和 Java 8 完美配合?
- 什么是适用于较新版本的等效项?
使用通用的网络搜索引擎让我对发生的事情一无所知,而且我能够找到的文档并未表明 setStatus() 只能被调用一次,也没有表明它可以被调用多次。
最佳答案
多次调用 setStatus()
并不被禁止,如果您查看 Tomcat 内部结构,您会发现有些地方可以多次更改状态(当然如果被禁止,你会得到一个异常(exception))。
这是由 regression bug 引起的在 Tomcat 9.0.10 和 9.0.9 中已修复,但在 9.0.8 中未修复(没有查看引入错误的位置,可能在 9.0.8 中)。
如果状态代码已经设置为超过 399 的值,则基本上尝试更改状态代码没有任何效果,因为
if (this.status > 399) {
// Don't overwrite first recorded error status
return;
}
关于java - HttpServletResponse.setStatus() 工作一次,再次调用时什么也不做 - 在 Java 8/Tomcat 9.0.0 和 Java 10/Tomcat 9.0.8 之间切换?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/51423223/