tomcat5下jsp出现getOutputStream() has already been called for t

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();
}%>

### 关于 `getOutputStream()` 被调用后的响应错误解决方案 当遇到 `java.lang.IllegalStateException: getOutputStream() has already been called for this response` 的异常时,通常意味着在同一请求处理过程中尝试多次获取同一响应的输出流或字符输出流。这违反了Servlet API的规定,在Java Web应用程序中是一个常见的编程陷阱。 #### 原因分析 该问题的根本原因是试图在一个HTTP响应对象上先后调用 `getWriter()` 和 `getOutputStream()` 方法,而这两个方法只能择一使用,并且一旦选择了其中一个,则不能再访问另一个[^1]。此外,如果在Spring框架环境中操作复杂的数据结构(如列表),并将其直接作为依赖注入到控制器或其他组件中,可能会间接引发此类冲突[^5]。 #### 解决策略 ##### 避免重复调用 确保在整个请求生命周期内仅调用一次 `getOutputStream()` 或者 `getWriter()` 。可以通过重构代码逻辑来实现这一点: ```java if (conditionForBinaryData) { try(OutputStream out = response.getOutputStream()) { // 使用try-with-resources语句自动关闭资源 byte[] data = ...; // 准备二进制数据 out.write(data); } } else { PrintWriter writer = response.getWriter(); String content = "Some text"; writer.print(content); } ``` ##### 处理复杂的业务场景 对于涉及大量计算或者需要返回动态生成文件的情况,考虑创建新的实例而不是共享同一个服务层的对象。特别是当这些对象包含了难以序列化的成员变量时,更应该如此做。例如,可以手动实例化所需的工具类而非依赖容器管理其生命周期。 ##### 清晰定义MIME类型 设置正确的Content-Type头可以帮助浏览器正确解析服务器发送过来的信息。如果不小心设置了不匹配的内容类型,也可能触发此异常。因此建议显式指定预期格式: ```java response.setContentType("application/pdf"); // 或者其他合适的媒体类型 ``` ##### 捕获潜在的竞争条件 有时并发环境下多个线程可能同时修改相同的HttpServletResponse实例,从而造成竞争状态。应采取措施防止这种情况发生,比如同步关键部分的操作或是利用ThreadLocal机制隔离各次请求间的上下文环境。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值