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

本文介绍在Tomcat5环境下,JSP中使用输出流导致getOutputStream()hasalreadybeencalledforthisresponse异常的原因及解决办法。通过正确处理输出流并提供一个输出彩色验证码的示例。

 

我自己也遇到了,这中问题,我用了第一种方法就解决了

1.

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

在tomcat5下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 i=0;i<4;i++){
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*i+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()放到循环体内 
### 问题分析 在Java Web应用中,`getOutputStream()` `getWriter()` 是 `HttpServletResponse` 接口中提供的两个方法,用于向客户端发送响应内容。`getOutputStream()` 通常用于输出二进制数据(如图片、文件等),而 `getWriter()` 用于输出字符文本(如HTML、JSON等)。根据Servlet规范,这两个方法不能同时使用。一旦其中一个方法被调用,再调用另一个方法将抛出 `IllegalStateException`,错误信息为: ``` getOutputStream() has already been called for this response ``` 此问题常见于JSP页面中,因为JSP默认会调用 `getWriter()` 来输出内容,如果在JSP中手动调用 `getOutputStream()`,就会发生冲突。 ### 解决方案 1. **避免在JSP中使用 `getOutputStream()`** JSP页面默认使用 `getWriter()` 输出内容,因此不建议在JSP中直接使用 `getOutputStream()`。如果需要输出二进制数据(如图片、文件等),应使用Servlet来处理此类请求。 2. **在Servlet中处理二进制响应** 如果需要输出二进制内容(如图片验证码、文件下载等),应在Servlet中完成操作,并确保不使用JSP页面来处理输出流。示例代码如下: ```java protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { response.setContentType("image/jpeg"); try (OutputStream out = response.getOutputStream()) { // 生成图片或读取文件并写入输出流 BufferedImage image = generateImage(); // 假设该方法生成图片 ImageIO.write(image, "JPEG", out); } } ``` 3. **关闭JSP的自动输出功能** 如果确实需要在JSP中使用 `getOutputStream()`,可以通过设置 `isELIgnored` `isThreadSafe` 属性来控制JSP的行为,并在JSP中显式关闭自动输出。例如: ```jsp <%@ page isELIgnored="false" buffer="none" autoFlush="false" %> ``` 然后在JSP中手动控制输出流: ```jsp <% response.setContentType("image/jpeg"); try (OutputStream out = response.getOutputStream()) { // 生成图片或读取文件并写入输出流 BufferedImage image = generateImage(); // 假设该方法生成图片 ImageIO.write(image, "JPEG", out); } %> ``` 4. **确保流正确关闭** 在使用完输出流后,应确保其被正确关闭,以避免资源泄漏或后续的冲突。可以在 `finally` 块中关闭流,或使用 try-with-resources 语法。 ```java try (OutputStream out = response.getOutputStream()) { // 输出内容 } catch (IOException e) { e.printStackTrace(); } ``` 5. **避免在JSP中混合使用 `out` `getOutputStream()`** JSP中的 `out` 对象本质上是通过 `getWriter()` 获取的字符输出流,与 `getOutputStream()` 冲突。因此,在使用 `getOutputStream()` 时,应避免使用 `out` 对象输出内容。 --- ###
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值