HttpServletResponse.setStatus() 工作一次,再次调用时不执行任何操作 - 在 Java 8/Tomcat 9.0.0 和 Java 10/Tomcat 9.0.8 之间更改?

a C*_*CVn 5 java tomcat servlets

我有一个在 Tomcat 9 下运行的 Java servlet,作为正常流程的一部分,它会调用HttpServletResponse#setStatus()几次。

当使用 Java 8(1.8.0u144,Tomcat 报告为1.8.0_144-b01)在 Tomcat 9.0.0.M26 上运行时,效果很好。

当在带有 Java 10.0.1 的 Tomcat 9.0.8.0 上运行时(Tomcat 报告为10.0.1+10),似乎在响应对象上调用 setStatus() 实际上只会导致响应状态被设置一次,此后 HTTP 状态将无法再设置改变了。然而,通过 HttpServletResponse#setHeader() 发送到客户端的其他标头似乎不受此影响;即使 setStatus() 不再执行任何操作后,setHeader() 也会成功添加标头。没有发送任何可能导致 HTTP 标头终止的中间输出数据。

这是一个最小的工作示例:

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()));
    }
}
Run Code Online (Sandbox Code Playgroud)

调用此 servlet 时,我希望返回字符串Testing Served at: ... with: 506,因为调用前设置的最后一个 HTTP 状态getStatus()506。返回给客户端的 HTTP 状态代码同样应该是506.

然而,我最终得到的是Testing Served at: ... with: 505HTTP505状态。就像第二个 setStatus() 调用根本不存在一样。

Testing无论是否包含 setStatus() 调用之前的结果都是相同的(除了response.getWriter().append("Testing ");输出开头存在 ),因此它似乎与提前终止 HTTP 响应标头无关。

没有任何迹象表明我可以看到第二个 setStatus() 调用以任何方式失败,甚至它曾经存在过;看起来除了第一次调用 setStatus() 之外,在响应对象上调用 setStatus() 绝对不会执行任何操作。

的返回值response.isCommitted()贯穿false于有问题的服务器上的上述 servlet:调用后getWriter().append("Testing ");、调用后setStatus(505)、调用后setStatus(506)

我意识到对同一个请求多次调用 setStatus() 可能有点不正统,但是:

  • 鉴于它与 Tomcat 9.0.0 和 Java 8 完美配合,这真的不应该与 Tomcat 9.0.8 和 Java 10 配合使用吗?
  • 与最新版本兼容的等效项是什么?

使用常见的网络搜索引擎让我对发生的事情一无所知,而且我找到的文档并不表明 setStatus() 只能调用一次,也不表明它可以调用多次。

Kay*_*man 3

多次调用setStatus()并不被禁止,如果你查看 Tomcat 内部结构,你会发现有些地方可以多次更改状态(当然,如果被禁止,你会得到一个异常)。

这是由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;
}
Run Code Online (Sandbox Code Playgroud)