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

}

}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/575773.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

c# 关于WebBrowser的模拟提交InvokeMember方法是什么意思!

一开始接触InvokeMember方法我以为他就是向页面插入javascript脚本!然后我想找系统帮我插入的这个脚本,不过找不到!,因为我不理解模拟这个词!哈哈 其实呢,不是这样子的! InvokeMember("cli…

java继续_Java中消除实现继续和面向接口编程

在匆忙之际理清消除实现继续和面向接口编程这样两个大题目可不是一件轻易的事情,尤其考虑到自身的熟悉水平。坦白的说,这又是一篇“炒冷饭”的文章,但这“冷饭”又确实不好炒。因此,在阅读了这篇文章之后,你可要批判地…

《转》VC++多线程编程

原地址:http://www.cnblogs.com/wxfasdic/archive/2010/09/23/1833522.html留个纪念,不错的总结。十个例子清晰列举啦多线程编程的奥妙。 VC中多线程使用比较广泛而且实用,在网上看到的教程.感觉写的挺好. 一、问题的提出编写一个耗时的单线程程序&#…

array函数参数 scala_scala – 在Spark SQL中将数组作为UDF参数传递

很可能不是最漂亮的解决方案,但你可以尝试这样的事情:def getCategory(categories: Array[String]) {udf((input:String) > categories(input.toInt))}df.withColumn("newCategory", getCategory(myArray)(col("myInput")))您还可以尝试一系…

Java数据类型BooleanDemo

转载于:https://www.cnblogs.com/suncoolcat/p/3320306.html

beetle.java 分析_Beetl模板引擎入门教程

最近项目中有个邮件发送的需求,不过要求发送的HTML格式的邮件。由于Beetl对java语言的良好支持和很好的性能,我们决定使用Beetl作为我们的模板引擎。Beetl官网已经有了很详细的教程,所以本篇侧重于实战应用,适合需要不懂beetl或其…

WebScoket 规范 + WebSocket 协议

WebSocket握手协议 1、客户端握手请求(注意:键值之间有一个空格,行间有换行符号0x13x10或者说\r\n) GET /WebSocket/LiveVideo HTTP/1.1 Upgrade: WebSocket Connection: Upgrade Host: localhost:8080 (客户端请求主机) Origin:…

heap python_数据结构-堆(Heap) Python实现

堆(Heap)可以看成近似完全二叉树的数组,树中每个节点对应数组中一个元素。除了最底层之外,该树是完全充满的,最底层是从左到右填充的。堆包括最大堆和最小堆:最大堆的每一个节点(除了根结点)的值不大于其父节点;最小堆…

多个 App 间启动

http://developer.nokia.com/: URI associations for Windows Phone http://msdn.microsoft.com/: Auto-launching apps using file and URI associations for Windows Phone 8 代码示例转载于:https://www.cnblogs.com/sirkevin/p/3325035.html

im4java 文档_im4java学习---阅读documentation文档

Utilities----im提供的一些工具类①、读取图片文件信息---Info类我们之前的做法:op.format("width:%w,height:%h,path:%d%f,size:%b%[EXIF:DateTimeOriginal]");IdentifyCmd identifyCmd new IdentifyCmd(useGM);使用工具类Info:Info imageIn…

函数体中定义的结构体和类型

源代码&#xff1a; 1 #include <stdio.h>2 struct smonth // point 13 {4 int a;5 int b;6 };7 8 int func1()9 { 10 struct smonth{ 11 int a; 12 int b; 13 }; 14 15 ty…

java listview用法_Java ListView.setMultiChoiceModeListener方法代码示例

import android.widget.ListView; //导入方法依赖的package包/类Overridepublic void onActivityCreated(Nullable Bundle savedInstanceState) {super.onActivityCreated(savedInstanceState);lAdapter new LabelAdapter(getActivity(), null, 0);setListAdapter(lAdapter);g…

预编译指令与宏定义

#if #elif [defined(), !defined()] #else #ifdef #ifndef #endif // 条件编译 /* 头文件防止多次被包含 */ #ifndef ZLIB_H #define ZLIB_H#endif /* ZLIB_H *//* 用C方式来修饰函数与变量 */ #ifdef __cplusplus extern "C" { #endif int add(int a, …

java mock server_java – 使用MockRestServiceServer模拟REST调用

我正在尝试编写一个JUnit测试用例,用于测试辅助类中的方法.该方法使用REST调用外部应用程序,这是我试图在JUnit测试中模拟的调用.辅助方法使用Spring的RestTemplate进行REST调用.在我的测试中,我创建了一个模拟REST服务器并模拟REST模板并将它们实例化为&#xff1a;Beforepubl…

BZOJ 2597 剪刀石头布(最小费用最大流)(WC2007)

Description 在一些一对一游戏的比赛&#xff08;如下棋、乒乓球和羽毛球的单打&#xff09;中&#xff0c;我们经常会遇到A胜过B&#xff0c;B胜过C而C又胜过A的有趣情况&#xff0c;不妨形象的称之为剪刀石头布情况。有的时候&#xff0c;无聊的人们会津津乐道于统计有多少这…

java swt最小化到托盘_SWT 中实现最小化到托盘图标,并只能通过托盘的弹出菜单关闭程序...

package com.unmi;import org.eclipse.swt.*;import org.eclipse.swt.events.*;import org.eclipse.swt.graphics.*;import org.eclipse.swt.widgets.*;/*** SWT 3.0 开始引入了 Tray&#xff0c;可以在系统栏放置你的程序图标了* 本程序实现的功能有四&#xff1a;* 1. 点击窗…

HDU 1476 Sudoku Killer

Sudoku Killer Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/32768 K (Java/Others) Total Submission(s): 3367 Accepted Submission(s): 1063 Problem Description自从2006年3月10日至11日的首届数独世界锦标赛以后&#xff0c;数独这项游戏越来越受到…

java .net 互通redis_C# servicestack.redis 互通 java jedis

本文是基于jedis的一致性环哈希来修改的&#xff0c;.net选的是servicestack.redis组件来修改无奈两个组件都有各自的一致性环哈希算法&#xff0c;不兼容&#xff0c;那就选一个作为标准&#xff0c;修改另一个咯。本文选择jedis的一致性环哈希作为标准&#xff0c;进而修改.n…

java 多重压缩下载_Java 多文件边压缩边下载

有时我们希望在后台实时生成文件并下载到客户端GetMapping(value "download")public void download(HttpServletResponse response) {try(OutputStream outputStream response.getOutputStream();ZipOutputStream zipOutputStream new ZipOutputStream(outputStre…

wdatepicker使用指南

wdatepicker使用指南 http://blog.csdn.net/zengxin2008/article/details/7248964#t63posted on 2013-09-22 15:45 moonfans 阅读(...) 评论(...) 编辑 收藏 转载于:https://www.cnblogs.com/moonfans/p/3333251.html