java response 输出流_java-springmvc+filter 替换输出流、response、响应内容
java-springmvc+filter 替換輸出流、response、響應內容
一、問題
1.描述:在使用 filter 替換、修改 response 輸出內容時常見的錯誤如下異常提示
getWriter() has already been called for this response
getOutputStream() has already been called for this response
2.問題產生原因:
getWriter() 和 getOutputStream() 方法互斥,一個 response 只允許調用一次;
getWriter() 對應一個字符流,用于處理純文本相關的資源;
getOutputStream() ?對應一個字節流,用于處理如圖片之類的資源;
3.解決辦法:
自定義一個包裝器繼承?HttpServletResponseWrapper 類,并且重寫以下兩個方法,且兩個方法都向同一個輸出流中寫入內容;
public PrintWriter getWriter();
public PrintWriter getOutputStream();
4.注意:有時訪問 jsp 頁面或其它內容時,沒有內容輸出。分析是不是沒有調用字節流、字符流的 flush() 方法。
二、下面使用 springmvc 的 OncePerRequestFilter 實現一個替換 response 內容的 filter;當然也可以直接實現 Filter 接口
1. web.xml 配置filter
AuthCodeFilter
com.demo.web.filter.AuthCodeFilter
enable
false
exclude_url
(/login\.jsp22)$|(\.css)$|(\.js)$|(\.jpg)$|(\.png)$|(\.gif)$|(\.pdf)$|(\.eot)$|(\.svg)$|(\.ttf)$|(\.woff)$|(\.woff2)$
content_type
(text/.+)
AuthCodeFilter
/*
REQUEST
FORWARD
INCLUDE
ERROR
2.AuthCodeFilter.java
package com.demo.web.filter;
import java.io.IOException;
import java.util.regex.Matcher;
import java.util.regex.Pattern;
import javax.servlet.FilterChain;
import javax.servlet.FilterConfig;
import javax.servlet.ServletException;
import javax.servlet.ServletOutputStream;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import com.demo.web.rules.sys.AuthRule;
import me.grass.coder.Debug;
import me.grass.extend.StringExtend;
/**
* 功能權限篩選器
* @author xxj
*/
public class AuthCodeFilter extends org.springframework.web.filter.OncePerRequestFilter{
Pattern _pattenUrl;
Pattern _pattenContentType;
boolean _enbale=true;
AuthRule _rule = AuthRule.instance();
@Override
protected void initFilterBean() throws ServletException {
FilterConfig conf = this.getFilterConfig();
String enable = conf.getInitParameter("enable");
String regex = conf.getInitParameter("exclude_url");
String regexContentType = conf.getInitParameter("content_type");
Debug.printFormat("{2} init-param: enable={0};exclude_url={1}",enable,regex,this.getClass().getName());
_pattenContentType = Pattern.compile(regexContentType, Pattern.CASE_INSENSITIVE);
_enbale = StringExtend.getBoolean(enable);
// 初始化正則驗證器
if(_pattenUrl==null){
//忽略大小寫
_pattenUrl = Pattern.compile(regex, Pattern.CASE_INSENSITIVE);
Debug.printFormat("{2}初始化;Enable={1};content-type正則:{3};url正則 ={0};"
, regex
,_enbale
,this.getClass().getSimpleName()
,regexContentType);
}
}
@Override
public void destroy() {
}
@Override
protected void doFilterInternal(HttpServletRequest request, HttpServletResponse response
, FilterChain filter)
throws ServletException, IOException {
//是否啟用篩選器
if (!_enbale) {
filter.doFilter(request, response);
return;
}
HttpServletRequest req = (HttpServletRequest) request;
String url = req.getRequestURI();
//1 處理 request 請求信息
//1.1 不驗證的資源
Matcher matcher = _pattenUrl.matcher(url);
if (matcher.find()) {
filter.doFilter(request, response);
return;
}
// 1.2 功能權限驗證
// 1.2.1 實例化一個響應包裝器,用于緩存 response 中的內容到 CharArrayWriter 對象中
AuthCodeResponseWrapper authResp = new AuthCodeResponseWrapper((HttpServletResponse) response);
// 2 調用 doFilter() 方法,繼續執行 filter 鏈中其它 filter
filter.doFilter(request, authResp);
// 3 處理 response 響應信息
ServletOutputStream out = response.getOutputStream();
// 3.1 不需要驗證的 content-type
String contentType = response.getContentType();
if(!StringExtend.isNullOrEmpty(contentType)){
matcher = _pattenContentType.matcher(contentType);
if(!matcher.find()){
authResp.getByteArrayOutputStream().writeTo(out);
return;
}
}
// 3.2 filter 鏈執行結束,獲取 CharArrayWriter 的內容
// 3.3 將 content 內容進行過濾
String content = authResp.getTextContent();
String html = content.replece("hello word!","你好,世界!"); //替換敏感詞
if(StringExtend.isNullOrEmpty(html)){
authResp.getByteArrayOutputStream().writeTo(out);
return;
}
// 3.4 將過濾后的內容寫入響應流中
if(!_rule.isFilter()){//沒有進行過功能篩選則原樣輸出
authResp.getByteArrayOutputStream().writeTo(out);
return;
}
//3.5 寫入輸出流
out.write(html.getBytes());
Debug.printFormat("[權限過濾] url={0}", url);
}
}
3.AuthCodeResponseWrapper.java
package com.demo.web.filter;
import java.io.ByteArrayOutputStream;
import java.io.IOException;
import java.io.PrintWriter;
import javax.servlet.ServletOutputStream;
import javax.servlet.http.HttpServletResponse;
import javax.servlet.http.HttpServletResponseWrapper;
import me.grass.coder.Debug;
/**
* 功能權限響應對象
* @author xxj
*/
public class AuthCodeResponseWrapper extends HttpServletResponseWrapper {
ByteArrayOutputStream _stream = new ByteArrayOutputStream();
PrintWriter _pw=new PrintWriter(_stream);
public AuthCodeResponseWrapper(HttpServletResponse response) {
super(response);
}
/**
* 覆蓋getWriter()方法,將字符流緩沖到本地
*/
@Override
public PrintWriter getWriter() throws IOException {
Debug.print("getWriter()");
return _pw;
}
/**
* 覆蓋getOutputStream()方法,將字節流緩沖到本地
*/
@Override
public ServletOutputStream getOutputStream() throws IOException {
Debug.print("getOutputStream()");
return new ServletOutputStream(){
@Override
public void write(int b) throws IOException {
_stream.write(b);
}
};
}
/**
* 把緩沖區內容寫入輸出流后關閉
*
* @author xxj
*/
public void flush(){
try {
_pw.flush();
_pw.close();
_stream.flush();
_stream.close();
} catch (IOException e) {
e.printStackTrace();
}
}
/**
* 獲取字節流
* @return
*/
public ByteArrayOutputStream getByteArrayOutputStream(){
return _stream;
}
/**
* 將換出區內容轉為文本輸出
* @return
*/
public String getTextContent() {
flush();
return _stream.toString();
}
}
總結
以上是生活随笔為你收集整理的java response 输出流_java-springmvc+filter 替换输出流、response、响应内容的全部內容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: 汇编语言标识符及其命名规则
- 下一篇: 第一个汇编语言程序