getOutputStream() has already been called for this response异常的原因和解决方法

1.tomcat6.0下jsp出现getOutputStream() has already been called for this response异常的原因和解决方法

  在tomcat6.0下jsp中出现此错误一般都是在jsp中使用了输出流(如输出图片验证码,文件下载等),没有妥善处理好的原因。

  具体的原因就是:
  在tomcat中jsp编译成servlet之后在函数_jspService(HttpServletRequest request, HttpServletResponse response)的最后有一段这样的代码

1 finally {
2       if (_jspxFactory != null) _jspxFactory.releasePageContext(_jspx_page_context);
3 }

  这里是在释放在jsp中使用的对象,会调用response.getWriter(),因为这个方法是和response.getOutputStream()相冲突的!所以会出现以上这个异常。

  然后当然是要提出解决的办法,其实挺简单的(并不是和某些朋友说的那样--将jsp内的所有空格和回车符号所有都删除掉),在使用完输出流以后调用以下两行代码即可:

1 out.clear();
2 out = pageContext.pushBody();

为何加这两句话呢?

out.clear(); 

out = pageContext.pushBody(); 
复制代码out = pageContext.pushBody(); 是什么意思?为什么不加入了就报异常: java.lang.IllegalStateException: getOutputStream() has already been called for this response 
复制代码 
首先你要知道pushBody()的作用是保存当前的out对象,并更新PageContext中Page范围内Out对象。至于为什么要加上这句话,是因为JSP容器在处理完成请求后会调用releasePageConter方法释放所有的PageContestObject,并且同时调用getWriter方法。由于getWriter方法与在JSP页面中使用流相关的getOutputStream方法冲突,所以会造成这种异常,解决方法就是楼上给的一样,只需要在JSP页面的最后加上这两条语句。out.clear(); 

out = pageContext.pushBody(); 
复制代码




   最后这里是一个输出彩色验证码例子(这样的例子几乎随处可见)

 imag.jsp

 1  <%@ page import="java.awt.*,java.awt.image.*,java.util.*,javax.imageio.*" %>
 2  <%@ page import="java.io.OutputStream" %>
 3  <%!
 4    Color getRandColor(int fc,int bc){
 5      Random random = new Random();
 6      if(fc>255) fc=255;
 7      if(bc>255) bc=255;
 8      int r=fc+random.nextInt(bc-fc);
 9      int g=fc+random.nextInt(bc-fc);
10      int b=fc+random.nextInt(bc-fc);
11      return new Color(r,g,b);
12    }
13  %>
14  <%
15    try{
16      response.setHeader("Pragma","No-cache");
17      response.setHeader("Cache-Control","no-cache");
18      response.setDateHeader("Expires", 0);
19      int width=60, height=20;
20      BufferedImage image = new BufferedImage(width, height, BufferedImage.TYPE_INT_RGB);
21      OutputStream os=response.getOutputStream();
22      Graphics g = image.getGraphics();
23      Random random = new Random();
24      g.setColor(getRandColor(200,250));
25      g.fillRect(0, 0, width, height);
26  
27      g.setFont(new Font("Times New Roman",Font.PLAIN,18));
28      g.setColor(getRandColor(160,200));
29      for (int i=0;i<155;i++){
30        int x = random.nextInt(width);
31        int y = random.nextInt(height);
32        int xl = random.nextInt(12);
33        int yl = random.nextInt(12);
34        g.drawLine(x,y,x+xl,y+yl);
35      }
36      String sRand="";
37      for (int j=0;j<4;j++){
38        String rand=String.valueOf(random.nextInt(10));
39        sRand+=rand;
40        g.setColor(new Color(20+random.nextInt(110),20+random.nextInt(110),20+random.nextInt(110)));
41        g.drawString(rand,13*j+6,16);
42      }
43      session.setAttribute("rand",sRand);
44      g.dispose();
45  
46      ImageIO.write(image, "JPEG",os);
47      os.flush();
48      os.close();
49      os=null;
50      response.flushBuffer();
51      out.clear();
52      out = pageContext.pushBody();
53    }catch(IllegalStateException e){
54       System.out.println(e.getMessage());
55      e.printStackTrace();
56    }
57   %>

  如有不足之处,欢迎斧正!

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)

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值