java - HttpServletResponse.setStatus() 工作一次,再次调用时什么也不做 - 在 Java 8/Tomcat 9.0.0 和 Java 10/Tomcat 9.0.8 之间切换?

标签 java tomcat servlets

我有一个在 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/

相关文章:

java - 从独立 JVM 访问 Tomcat 8 JNDI

security - 在 web.xml 之外定义一个 <security-constraint>(例如,服务器范围)

forms - 成功登录Tomcat表单后重定向

java - 使用 JNI 从 Java 调用 C++ 程序时出错。获取 SEGV_ACCERR

tomcat - 在同一 tomcat 但不同端口上部署 Web 服务和 Web 门户时的 JVM

java - 应用程序不在服务器上生成报告

java - tomcat是如何解析资源的?

java - 从文件加载大型 2D int 数组的最快方法是什么?

java - 替换 jpa 中的月、年、日期函数

java - 为什么此代码显示参数索引超出范围异常?