解决getOutputStream() has already been called for this response

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

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


具体的原因:jsp编译成servlet之后在函数

_jspService(HttpServletRequest request, HttpServletResponse response)

的最后
有一段这样的代码

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

这里是在释放在jsp中使用的对象,会调用response.getWriter(),因为这个方法是和response.getOutputStream()相冲突的!所以会出现以上这个异常。然后当然是要提出解决的办法,其实挺简单的,在使用完输出流以后调用以下两行代码即可:

Java代码
  1. out.clear();
  2. out = pageContext.pushBody();
out.clear();
out = pageContext.pushBody();

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

Java代码
  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. g.setFont(new Font("Times New Roman",Font.PLAIN,18));
  27. g.setColor(getRandColor(160,200));
  28. for (int i=0;i<155;i++)
  29. {
  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 i=0;i<4;i++){
  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*i+6,16);
  42. }
  43. session.setAttribute("rand",sRand);
  44. g.dispose();
  45. ImageIO.write(image, "JPEG",os);
  46. //注意看以下几句的使用
  47. os.flush();
  48. os.close();
  49. os=null;
  50. response.flushBuffer();
  51. out.clear();
  52. out = pageContext.pushBody();
  53. }
  54. catch(IllegalStateException e)
  55. {
  56. System.out.println(e.getMessage());
  57. e.printStackTrace();
  58. }%>
<%@ 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();
}%>


如果写文件是在java类中实现,可参考如下代码:

Java代码
  1. FileOutputStream out = new FileOutputStream(file,true);
  2. out.write(new byte[]{(byte)0xEF, (byte)0xBB, (byte)0xBF});//utf-8 bom
  3. out.write(content.getBytes(charset));
  4. out.close();
FileOutputStream out = new FileOutputStream(file,true);
out.write(new byte[]{(byte)0xEF, (byte)0xBB, (byte)0xBF});//utf-8 bom
out.write(content.getBytes(charset));
out.close();

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
当你在项目中进行数据导出到Excel时,你可能会遇到"getOutputStream() has already been called for this response"的异常。这个异常的原因可能是因为在你的代码中,同时调用了response.getOutputStream()和response.getWriter(),而根据Servlet规范,这两个方法不能同时调用。 在你提供的代码中,我可以看到你使用了@GetMapping注解来处理导出功能,并在方法参数中传入了HttpServletResponse对象。然后,你调用了ExcelPortUtil.excelPort()方法来进行数据写入Excel文件并返回。 为了解决"getOutputStream() has already been called for this response"异常,你可以按照下面的步骤进行调整: 1. 确保在你的代码中只调用response.getOutputStream()或response.getWriter()其中之一。 2. 检查你的ExcelPortUtil.excelPort()方法内部的实现,确保没有调用response.getOutputStream()或response.getWriter()。 3. 如果你的ExcelPortUtil.excelPort()方法内部确实需要使用response.getOutputStream(),那么你需要修改你的方法逻辑,确保在调用response.getOutputStream()之前没有调用response.getWriter()。 通过以上调整,你应该能够解决"getOutputStream() has already been called for this response"的异常。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* *2* *3* [getOutputStream() has already been called for this response问题终极解决](https://blog.csdn.net/IT_51888_liang/article/details/122235797)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 100%"] [ .reference_list ]

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

连敬磊

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值