106

这个方法抛出

java.lang.IllegalStateException:提交响应后无法转发

我无法发现问题。有什么帮助吗?

    int noOfRows = Integer.parseInt(request.getParameter("noOfRows"));
    String chkboxVal = "";
    // String FormatId=null;
    Vector vRow = new Vector();
    Vector vRow1 = new Vector();
    String GroupId = "";
    String GroupDesc = "";
    for (int i = 0; i < noOfRows; i++) {
        if ((request.getParameter("chk_select" + i)) == null) {
            chkboxVal = "notticked";
        } else {
            chkboxVal = request.getParameter("chk_select" + i);
            if (chkboxVal.equals("ticked")) {
                fwdurl = "true";
                Statement st1 = con.createStatement();
                GroupId = request.getParameter("GroupId" + i);
                GroupDesc = request.getParameter("GroupDesc" + i);
                ResultSet rs1 = st1
                        .executeQuery("select FileId,Description from cs2k_Files "
                                + " where FileId like 'M%' and co_code = "
                                + ccode);
                ResultSetMetaData rsm = rs1.getMetaData();
                int cCount = rsm.getColumnCount();

                while (rs1.next()) {
                    Vector vCol1 = new Vector();
                    for (int j = 1; j <= cCount; j++) {
                        vCol1.addElement(rs1.getObject(j));
                    }
                    vRow.addElement(vCol1);
                }
                rs1 = st1
                        .executeQuery("select FileId,NotAllowed from cs2kGroupSub "
                                + " where FileId like 'M%' and GroupId = '"
                                + GroupId + "'" + " and co_code = " + ccode);
                rsm = rs1.getMetaData();
                cCount = rsm.getColumnCount();

                while (rs1.next()) {
                    Vector vCol2 = new Vector();
                    for (int j = 1; j <= cCount; j++) {
                        vCol2.addElement(rs1.getObject(j));
                    }
                    vRow1.addElement(vCol2);
                }

                // throw new Exception("test");

                break;
            }
        }
    }
    if (fwdurl.equals("true")) {
        // throw new Exception("test");
        // response.sendRedirect("cs2k_GroupCopiedUpdt.jsp") ;
        request.setAttribute("GroupId", GroupId);
        request.setAttribute("GroupDesc", GroupDesc);
        request.setAttribute("vRow", vRow);
        request.setAttribute("vRow1", vRow1);
        getServletConfig().getServletContext().getRequestDispatcher(
                "/GroupCopiedUpdt.jsp").forward(request, response);
    }
4

8 回答 8

264

初学者之间的一个常见误解是,他们认为调用 a forward(), sendRedirect(), orsendError()会神奇地退出并“跳”出方法块,从而忽略了代码的残余。例如:

protected void doXxx() {
    if (someCondition) {
        sendRedirect();
    }
    forward(); // This is STILL invoked when someCondition is true!
}

因此这实际上是不正确的。它们的行为肯定与任何其他 Java 方法没有什么不同(System#exit()当然是预期的)。当someCondition上面的示例是true并且您因此在相同的请求/响应forward()之后sendRedirect()或之后调用时,那么您将获得异常的机会很大sendError()

java.lang.IllegalStateException:提交响应后无法转发

如果该if语句调用 aforward()并且您随后调用sendRedirect()or sendError(),则将引发以下异常:

java.lang.IllegalStateException:提交响应后无法调用 sendRedirect()

要解决此问题,您需要在return;之后添加一个语句

protected void doXxx() {
    if (someCondition) {
        sendRedirect();
        return;
    }
    forward();
}

...或引入一个 else 块。

protected void doXxx() {
    if (someCondition) {
        sendRedirect();
    } else {
        forward();
    }
}

要确定代码中的根本原因,只需搜索任何调用 a 的行forward()sendRedirect()或者sendError()不退出方法块或跳过代码的剩余部分。这可以在特定代码行之前的同一 servlet 中,也可以在特定 servlet 之前调用的任何 servlet 或过滤器中。

如果是sendError(),如果您的唯一目的是设置响应状态,请setStatus()改用。


另一个可能的原因是 servlet 在 aforward()将被调用时写入响应,或者已经以相同的方法调用。

protected void doXxx() {
    out.write("some string");
    // ... 
    forward(); // Fail!
}

大多数服务器的响应缓冲区大小默认为 2KB,因此如果您向其写入超过 2KB,那么它将被提交并forward()以同样的方式失败:

java.lang.IllegalStateException:提交响应后无法转发

解决方案很明显,只是不要写入 servlet 中的响应。这是 JSP 的职责。您只需像这样设置一个请求属性request.setAttribute("data", "some string"),然后像这样在 JSP 中打印它${data}。另请参阅我们的 Servlets wiki 页面,了解如何正确使用 Servlets。


另一个可能的原因是 servlet 将文件下载写入响应,然后forward()调用例如 a。

protected void doXxx() {
    out.write(bytes);
    // ... 
    forward(); // Fail!
}

这在技术上是不可能的。您需要删除forward()呼叫。最终用户将停留在当前打开的页面上。如果您确实打算在文件下载后更改页面,那么您需要将文件下载逻辑移动到目标页面的页面加载。


另一个可能的原因是forward(),sendRedirect()sendError()方法是通过嵌入在 JSP 文件中的 Java 代码以老式方式调用的<% scriptlets %>,这种做法自 2001 年以来已被官方禁止。例如:

<!DOCTYPE html>
<html lang="en">
    <head>
        ... 
    </head>
    <body>
        ...

        <% sendRedirect(); %>
        
        ...
    </body>
</html>

out.write("<!DOCTYPE html> ... etc ...")这里的问题是,JSP 内部一旦遇到模板文本(即 HTML 代码)就会立即写入。因此,这与上一节中解释的问题基本相同。

解决方案很明显,只是不要在 JSP 文件中编写 Java 代码。这是普通 Java 类(如 Servlet 或 Filter)的职责。另请参阅我们的 Servlets wiki 页面,了解如何正确使用 Servlets。


也可以看看:


与您的具体问题无关,您的 JDBC 代码正在泄漏资源。也解决这个问题。有关提示,另请参阅在 JDBC 中应多久关闭一次连接、语句和结果集?

于 2010-01-23T22:51:07.430 回答
20

即使添加 return 语句也会引发此异常,对此唯一的解决方案是以下代码:

if(!response.isCommitted())
// Place another redirection
于 2013-02-23T19:44:02.800 回答
6

通常,您在完成重定向后会看到此错误,然后尝试将更多数据输出到输出流。在我过去看到过这种情况的情况下,它通常是试图重定向页面的过滤器之一,然后仍然转发到 servlet。我看不出 servlet 有什么问题,所以你可能想试试看一下你已经安装的任何过滤器。

编辑:诊断问题的更多帮助......</p>

诊断此问题的第一步是确定引发异常的确切位置。我们假设它被线路抛出

getServletConfig().getServletContext()
                  .getRequestDispatcher("/GroupCopiedUpdt.jsp")
                  .forward(request, response);

但是您可能会发现它稍后在代码中被抛出,您在尝试执行转发之后尝试输出到输出流。如果它来自上述行,则意味着在此行之前的某处您有:

  1. 将数据输出到输出流,或
  2. 事先做了另一个重定向。

祝你好运!

于 2010-01-23T14:50:25.887 回答
2

您应该在转发或重定向流时添加return语句。

例子:

如果转发,

    request.getRequestDispatcher("/abs.jsp").forward(request, response);
    return;

如果重定向,

    response.sendRedirect(roundTripURI);
    return;
于 2012-08-19T07:16:00.893 回答
2

这是因为您的 servlet 试图访问不再存在的请求对象。servlet 的 forward 或 include 语句不会停止方法块的执行。它继续到方法块的末尾或第一个返回语句,就像任何其他 java 方法一样。

解决此问题的最佳方法只是根据您的逻辑动态设置页面(您希望转发请求的位置)。那是:

protected void doPost(request , response){
String returnPage="default.jsp";
if(condition1){
 returnPage="page1.jsp";
}
if(condition2){
   returnPage="page2.jsp";
}
request.getRequestDispatcher(returnPage).forward(request,response); //at last line
}

并且在最后一行只做一次前锋......

您还可以在每个 forward() 之后使用 return 语句来解决此问题,或者将每个 forward() 放在 if...else 块中

于 2014-11-20T17:59:18.767 回答
2

我删除了

        super.service(req, res);

然后它对我来说很好

于 2015-08-14T07:21:16.500 回答
2

撞...

我只是有同样的错误。我注意到我super.doPost(request, response);在重写doPost()方法以及显式调用超类构造函数时正在调用

    public ScheduleServlet() {
        super();
        // TODO Auto-generated constructor stub
    }

一旦我评论了super.doPost(request, response);from insidedoPost()声明,它就完美地工作了......

protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {

        //super.doPost(request, response);
        // More code here...

}

不用说,我需要重新阅读super()最佳实践:p

于 2016-08-12T15:58:12.987 回答
0

在返回转发方法之后,您可以简单地执行以下操作:

return null;

它将打破当前范围。

于 2018-05-26T09:35:56.903 回答