日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 编程语言 > c/c++ >内容正文

c/c++

java response 输出流_java-springmvc+filter 替换输出流、response、响应内容

發布時間:2025/3/15 c/c++ 33 豆豆
生活随笔 收集整理的這篇文章主要介紹了 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、响应内容的全部內容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。