tomcat6下jsp出現getOutputStream() has already been called for this response異常的原因和解決方案_JSP編程

來源:互聯網
上載者:User

1.在tomcat6.0下jsp出現getOutputStream() has already been called for this response異常的原因和解決方案

  在tomcat6.0下jsp中出現此錯誤一般都是在jsp中使用了輸出資料流(如輸出圖片驗證碼,檔案下載等),沒有妥善處理好的原因。

  具體的原因就是:
  在tomcat中jsp編譯成servlet之後在函數_jspService(HttpServletRequest request, HttpServletResponse response)的最後有一段這樣的代碼

複製代碼 代碼如下:
finally {
    if (_jspxFactory != null) _jspxFactory.releasePageContext(_jspx_page_context);
}

  這裡是在釋放在jsp中使用的對象,會調用response.getWriter(),因為這個方法是和response.getOutputStream()相衝突的!所以會出現以上這個異常。

  然後當然是要提出解決的辦法,其實挺簡單的(並不是和部分朋友說的那樣--將jsp內的所有空格和斷行符號符號所有都刪除掉),在使用完輸出資料流以後調用以下兩行代碼即可:

複製代碼 代碼如下:
out.clear();
out = pageContext.pushBody();

  最後這裡是一個輸出彩色驗證碼例子(這樣的例子幾乎隨處可見)

imag.jsp

<%@ page import="java.awt.*,java.awt.image.*,java.util.*,javax.imageio.*" %><%@ page import="java.io.OutputStream" %><%!  Color getRandColor(int fc,int bc){    Random random = new Random();    if(fc>255) fc=255;    if(bc>255) bc=255;    int r=fc+random.nextInt(bc-fc);    int g=fc+random.nextInt(bc-fc);    int b=fc+random.nextInt(bc-fc);    return new Color(r,g,b);  } %><%  try{    response.setHeader("Pragma","No-cache");    response.setHeader("Cache-Control","no-cache");    response.setDateHeader("Expires", 0);    int width=60, height=20;    BufferedImage image = new BufferedImage(width, height, BufferedImage.TYPE_INT_RGB);    OutputStream os=response.getOutputStream();    Graphics g = image.getGraphics();    Random random = new Random();    g.setColor(getRandColor(200,250));    g.fillRect(0, 0, width, height);    g.setFont(new Font("Times New Roman",Font.PLAIN,18));    g.setColor(getRandColor(160,200));    for (int i=0;i<155;i++){      int x = random.nextInt(width);      int y = random.nextInt(height);      int xl = random.nextInt(12);      int yl = random.nextInt(12);      g.drawLine(x,y,x+xl,y+yl);     }    String sRand="";    for (int j=0;j<4;j++){      String rand=String.valueOf(random.nextInt(10));      sRand+=rand;      g.setColor(new Color(20+random.nextInt(110),20+random.nextInt(110),20+random.nextInt(110)));      g.drawString(rand,13*j+6,16);    }    session.setAttribute("rand",sRand);    g.dispose();    ImageIO.write(image, "JPEG",os);    os.flush();    os.close();    os=null;    response.flushBuffer();    out.clear();    out = pageContext.pushBody();  }catch(IllegalStateException e){      System.out.println(e.getMessage());    e.printStackTrace();  }%>

  如有不足之處,歡迎斧正!

2.getOutputStream() has already been called for this response問題的解決

  在jsp向頁面輸出圖片的時候,使用response.getOutputStream()會有這樣的提示:java.lang.IllegalStateException:getOutputStream() has already been called for this response,會拋出Exception

  原因一:
  JSP預設的輸出資料流為PrintWriter ,即<% %>以外的東西所預設的輸出方式,如果你嘗試在JSP中使用ServletOutputStream就會引起錯誤.要嘛直接改用Servlet輸出(複寫service方法),要嘛刪除除%><%中的任何東西(包括HTML標籤,空格,斷行符號等東西)應該就可以。對於這樣的情況應該這樣來解決,刪除%><%之間的所有內容包括空格和分行符號,最後也要消除空格和分行符號,最好再加上一句response.reset()。

  原因二: 

  在J2EE的API參考裡有這麼個:

  ServletResponse的getWriter()方法裡會拋出這個異常:

    IllegalStateException - if the getOutputStream method has already been called for this response object

  而它的getOutputStream()方法裡會拋出這個異常:

    IllegalStateException - if the getOutputStream method has already been called for this response object

  並且兩者的函數申明裡都有這麼樣的一句

    Either this method or getOutputStream() may be called to write the body, not both.
    Either this method or getWriter() may be called to write the body, not both.


  以上說明也解釋了為什麼在往頁面中寫入圖片的時候要使用如下迴圈格式

複製代碼 代碼如下:
OutputStream output=response.getOutputStream();
while((len=in.read(b)) >0) {
  output.write(b,0,len);
}
output.flush();

而不是把response.getOutputStream().write()放到迴圈體內

在頁面中直接寫:

複製代碼 代碼如下:
<body bgcolor="#ffffff">
<h1>
<%
    response.getOutputStream();
%>
</h1>
</body>

將會出現錯誤訊息如下:

java.lang.IllegalStateException: getOutputStream() has already been called for this response
org.apache.catalina.connector.Response.getWriter(Response.java:604)
org.apache.catalina.connector.ResponseFacade.getWriter(ResponseFacade.java:198)
org.apache.jasper.runtime.JspWriterImpl.initOut(JspWriterImpl.java:125)
org.apache.jasper.runtime.JspWriterImpl.flushBuffer(JspWriterImpl.java:118)

以上就是tomcat6下jsp出現getOutputStream() has already been called for this response異常的原因和解決方案的全部內容,希望能給大家一個參考,也希望大家多多支援雲棲社區。

相關文章

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.