tomcat6下jsp出現(xiàn)getOutputStream() has already been called for this response異常的原因和解決方法
1.在tomcat6.0下jsp出現(xiàn)getOutputStream() has already been called for this response異常的原因和解決方法
在tomcat6.0下jsp中出現(xiàn)此錯誤一般都是在jsp中使用了輸出流(如輸出圖片驗(yàn)證碼,文件下載等),沒有妥善處理好的原因。
具體的原因就是:
在tomcat中jsp編譯成servlet之后在函數(shù)_jspService(HttpServletRequest request, HttpServletResponse response)的最后有一段這樣的代碼
if (_jspxFactory != null) _jspxFactory.releasePageContext(_jspx_page_context);
}
這里是在釋放在jsp中使用的對象,會調(diào)用response.getWriter(),因?yàn)檫@個(gè)方法是和response.getOutputStream()相沖突的!所以會出現(xiàn)以上這個(gè)異常。
然后當(dāng)然是要提出解決的辦法,其實(shí)挺簡單的(并不是和某些朋友說的那樣--將jsp內(nèi)的所有空格和回車符號所有都刪除掉),在使用完輸出流以后調(diào)用以下兩行代碼即可:
out = pageContext.pushBody();
最后這里是一個(gè)輸出彩色驗(yàn)證碼例子(這樣的例子幾乎隨處可見)
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向頁面輸出圖片的時(shí)候,使用response.getOutputStream()會有這樣的提示:java.lang.IllegalStateException:getOutputStream() has already been called for this response,會拋出Exception
原因一:
JSP默認(rèn)的輸出流為PrintWriter ,即<% %>以外的東西所默認(rèn)的輸出方式,如果你嘗試在JSP中使用ServletOutputStream就會引起錯誤.要嘛直接改用Servlet輸出(復(fù)寫service方法),要嘛刪除除%><%中的任何東西(包括HTML標(biāo)簽,空格,回車等東西)應(yīng)該就可以。對于這樣的情況應(yīng)該這樣來解決,刪除%><%之間的所有內(nèi)容包括空格和換行符,最后也要消除空格和換行符,最好再加上一句response.reset()。
原因二:
在J2EE的API參考里有這么個(gè):
ServletResponse的getWriter()方法里會拋出這個(gè)異常:
IllegalStateException - if the getOutputStream method has already been called for this response object
而它的getOutputStream()方法里會拋出這個(gè)異常:
IllegalStateException - if the getOutputStream method has already been called for this response object
并且兩者的函數(shù)申明里都有這么樣的一句
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.
以上說明也解釋了為什么在往頁面中寫入圖片的時(shí)候要使用如下循環(huán)格式
while((len=in.read(b)) >0) {
output.write(b,0,len);
}
output.flush();
而不是把response.getOutputStream().write()放到循環(huán)體內(nèi)
在頁面中直接寫:
<h1>
<%
response.getOutputStream();
%>
</h1>
</body>
將會出現(xiàn)錯誤消息如下:
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出現(xiàn)getOutputStream() has already been called for this response異常的原因和解決方法的全部內(nèi)容,希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
相關(guān)文章
Servlet網(wǎng)上售票問題引發(fā)線程安全問題的思考
這篇文章主要是關(guān)于Servlet模擬網(wǎng)上售票問題,引發(fā)的線程安全問題的思考,感興趣的小伙伴們可以參考一下2015-12-12struts2+jsp實(shí)現(xiàn)文件上傳的方法
這篇文章主要介紹了struts2+jsp實(shí)現(xiàn)文件上傳的方法,涉及JSP基于Struts架構(gòu)實(shí)現(xiàn)文件傳輸?shù)耐暾记?具有一定參考借鑒價(jià)值,需要的朋友可以參考下2015-10-10JSP實(shí)現(xiàn)網(wǎng)頁訪問統(tǒng)計(jì)
這篇文章主要介紹了JSP實(shí)現(xiàn)網(wǎng)頁訪問統(tǒng)計(jì)方法,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2015-10-10jsp中一個(gè)頁面引入另一個(gè)頁面的實(shí)現(xiàn)代碼
這篇文章主要為大家分享了如果在jsp頁面中引入另一個(gè)頁面,需要的朋友可以參考下2013-11-11運(yùn)用JSP+ajax實(shí)現(xiàn)分類查詢功能的實(shí)例代碼
本篇文章主要介紹了運(yùn)用JSP+ajax實(shí)現(xiàn)分類查詢功能的實(shí)例代碼,具有一定的參考價(jià)值,有興趣的可以了解一下2017-07-07