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: 505
HTTP505
状态。就像第二个 setStatus() 调用根本不存在一样。
Testing
无论是否包含 setStatus() 调用之前的结果都是相同的(除了response.getWriter().append("Testing ");
输出开头存在 ),因此它似乎与提前终止 HTTP 响应标头无关。
没有任何迹象表明我可以看到第二个 setStatus() 调用以任何方式失败,甚至它曾经存在过;看起来除了第一次调用 setStatus() 之外,在响应对象上调用 setStatus() 绝对不会执行任何操作。
的返回值response.isCommitted()
贯穿false
于有问题的服务器上的上述 servlet:调用后getWriter().append("Testing ");
、调用后setStatus(505)
、调用后setStatus(506)
。
我意识到对同一个请求多次调用 setStatus() 可能有点不正统,但是:
使用常见的网络搜索引擎让我对发生的事情一无所知,而且我找到的文档并不表明 setStatus() 只能调用一次,也不表明它可以调用多次。
多次调用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)
归档时间: |
|
查看次数: |
7330 次 |
最近记录: |