利用HttpServletResponseWrapper对reponse进行包装

2024-06-07 13:32

本文主要是介绍利用HttpServletResponseWrapper对reponse进行包装,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

利用HttpServletResponseWrapper包装reponse

题记: 很多人或许用Spring的时候统一修改一下reponse信息,比较常见的就是给所有的response加一个status、code、message之类的要求,也有的想把response和request的日志打印出来,那么该如何做呢?那就耐心看看吧

一、背景介绍

  对于上述我们说到的这个需求,有很多中实现方式,比较容易想到的就是拦截器,过滤器等,这里我使用的是过滤器的方式来实现的,一是因为Spring天然就对Filter支持得比较好,二是开发方便容易。

  对于过滤器的实现方式,我这里也提供2种方式来实现,一种是复制ServletOutputStream的方式,一种是不复制ServletOutputStream的方式,两种方式各有用处,随君选择,后续我会开一篇原理的文章

  这里可能有读者疑问了,ServletOutputStream 是什么呀?我们看一下官方文档

Provides an output stream for sending binary data to the client. A ServletOutputStream object is normally retrieved via the ServletResponse.getOutputStream method.
This is an abstract class that the servlet container implements. Subclasses of this class must implement the java.io.OutputStream.write(int) method.

  文档里面写得很清楚,说提供一个输出流来向客户端发送数据,一个ServletOutputStream通常是reponse.getOutputStream()方法来得到,子类必须实现write方法。OK,说明我们最后要改的就是这个了,且要通过reponse.getOutputStream()这种方法得到。

二、不复制输出流方式(推荐,简便)

  1. 首先我们要创建一个Filter,这个Filter继承Spring的`OncePerRequestFilter

     继承这个Filter 跟继承普通的Filter主要区别就是它只会执行一次,在Servlet 3.0 之后,一个Filter可以被多个线程唤醒,这样就可能出现一些问题,所以我们这里使用这个

    参考文档:

    The dispatcher type javax.servlet.DispatcherType.ASYNC introduced in Servlet 3.0 means a filter can be invoked in more than one thread over the course of a single request. Some filters only need to filter the initial thread (e.g. request wrapping) while others may need to be invoked at least once in each additional thread for example for setting up thread locals or to perform final processing at the very end.
  2. ResponseWrapperFilter 代码如下

    @Order(Ordered.LOWEST_PRECEDENCE - 1)
    @WebFilter(urlPatterns = {"/common/*"},filterName = "responseWrapperFilter")
    public class ResponseWrapperFilter extends OncePerRequestFilter {private static final Logger LOGGER = LoggerFactory.getLogger(ResponseWrapperFilter.class);@Overrideprotected void doFilterInternal(HttpServletRequest request, HttpServletResponse response, FilterChain filterChain) throws ServletException, IOException {ResponseWrapper wrapper = new ResponseWrapper(response);try {filterChain.doFilter(request,wrapper);String responseStr = new String(wrapper.toByteArray(), response.getCharacterEncoding());Object parse = JSON.parse(responseStr);BaseResult<Object> baseResult = new BaseResult<>();baseResult.setData(parse);if (parse instanceof JSONObject){JSONObject resultObject = (JSONObject) parse;if (resultObject.containsKey("status")&&resultObject.containsKey("message")&&resultObject.containsKey("data")){baseResult = JSONObject.parseObject(resultObject.toJSONString(),new TypeReference<BaseResult<Object>>(){});}}LOGGER.info("response is ============{}",baseResult);//必须设置ContentLengthresponse.setContentLength(JSON.toJSONBytes(baseResult).length);//根据http accept来设置,我这里为了简便直接写json了response.setContentType("application/json;charset=utf-8");response.getOutputStream().write(JSON.toJSONBytes(baseResult));} catch (Exception e) {LOGGER.error("数据包装器执行出错....{}", e);}}
    }

    我这个Filter主要作用就是给我们的Reponse加上status,把Controller里面的数据放到data字段里面去,加一个包装。

  3. ResponseWrapper类源代码如下:

    private ByteArrayOutputStream output;
    private ServletOutputStream filterOutput;public ResponseWrapper(HttpServletResponse response) {super(response);output = new ByteArrayOutputStream();
    }/**
    * 巧妙将ServletOutputStream放到公共变量,解决不能多次读写问题
    * @return
    * @throws IOException
    */
    @Override
    public ServletOutputStream getOutputStream() throws IOException {if (filterOutput == null) {filterOutput = new ServletOutputStream() {@Overridepublic void write(int b) throws IOException {output.write(b);}@Overridepublic boolean isReady() {return false;}@Overridepublic void setWriteListener(WriteListener writeListener) {}};}return filterOutput;
    }public byte[] toByteArray() {return output.toByteArray();
    }
  4. BaseResult类

    public class BaseResult<T> implements Serializable {private static final long serialVersionUID = 2120267584344923858L;private Integer status = 0;private String message = null;private T data = null;public BaseResult(){}public BaseResult(Integer status, String message, T data) {this.status = status;this.message = message;this.data = data;}public Integer getStatus() {return status;}public void setStatus(Integer status) {this.status = status;}public String getMessage() {return message;}public void setMessage(String message) {this.message = message;}public T getData() {return data;}public void setData(T data) {this.data = data;}@Overridepublic String toString() {return "Result{" +"status=" + status +", message='" + message + '\'' +", data=" + data +'}';}
    }

  5. 这两个类与现有的Spring项目集成就可以实现将reponse修改的目的了

  6. 使用,浏览器访问:localhost:8080/common/getBaseResult 看效果

    {
    data: {
    imgUrl: "htt://img.baidu.com/images/11.jpg",
    name: "itar"
    },
    status: 0
    }
  7. 是不是还漏了一个Controller

    @RestController
    @RequestMapping({"/common","/never"})
    public class CommonControllerImpl{@RequestMapping(value = "/getBaseResult",method = RequestMethod.GET,produces = MediaType.APPLICATION_JSON_UTF8_VALUE) @ResponseBody public JSONObject getBaseResult() {JSONObject jsonObject = new JSONObject();jsonObject.put("imgUrl","htt://img.baidu.com/images/11.jpg");jsonObject.put("name","itar");return jsonObject;}
    }

三、复制输出流方式

  1. 我们这里的复制输出流主要就是将ServletOutputStream 这个流复制一份,复制出来之后大家可以写写日志呀,做些特殊处理什么的。

  2. 主要使用的是apache提供的TeeOutputStream类将输出流复制一份出来。

  3. BranchResponseWrapper 类源代码如下

    public class BranchResponseWrapper extends HttpServletResponseWrapper {/*** 我们的分支流*/private ByteArrayOutputStream output;private ServletOutputStream filterOutput;private OutputStream bufferOutputStream;public BranchResponseWrapper(HttpServletResponse response) {super(response);/*try {bufferOutputStream = response.getOutputStream();} catch (IOException e) {e.printStackTrace();}*/output = new ByteArrayOutputStream();}/*** 利用TeeOutputStream复制流,解决多次读写问题* 用super.getOutputStream来获取源outputstream,也可以用注释的那种方式获取,传过来* @return* @throws IOException*/@Overridepublic ServletOutputStream getOutputStream() throws IOException {if (filterOutput == null) {filterOutput = new ServletOutputStream() {//替换构造方法//拿父类的response,初始化的时候,里面还没有数据,只有一些request信息和response信息,但是调用了创建outputStream,//private TeeOutputStream teeOutputStream = new TeeOutputStream(bufferOutputStream,output);private TeeOutputStream teeOutputStream = new TeeOutputStream(BranchResponseWrapper.super.getOutputStream(),output);@Overridepublic boolean isReady() {return false;}@Overridepublic void setWriteListener(WriteListener writeListener) {}@Overridepublic void write(int b) throws IOException {teeOutputStream.write(b);}};}return filterOutput;}public byte[] toByteArray() {return output.toByteArray();}
    }
    
  4. BranchResponseWrapperFilter源代码如下

    @Order(Ordered.LOWEST_PRECEDENCE - 1)
    @WebFilter(urlPatterns = {"/never/*"},filterName = "branchResponseWrapperFilter")
    public class BranchResponseWrapperFilter extends OncePerRequestFilter {private static final Logger LOGGER = LoggerFactory.getLogger(BranchResponseWrapperFilter.class);@Overrideprotected void doFilterInternal(HttpServletRequest request, HttpServletResponse response, FilterChain filterChain) throws ServletException, IOException {BranchResponseWrapper wrapper = new BranchResponseWrapper(response);try {filterChain.doFilter(request,wrapper);String respStr= new String(wrapper.toByteArray(), response.getCharacterEncoding());Object parse = JSON.parse(respStr);JSONObject jsonObject = new JSONObject();jsonObject.put("status",0);jsonObject.put("data",parse);LOGGER.info("response is ============{}",jsonObject);response.setContentType("application/json;charset=utf-8");//将buffer重置,因为我们要重新写入流进去response.resetBuffer();response.setContentLength(JSON.toJSONBytes(jsonObject).length);response.getOutputStream().write(JSON.toJSONBytes(jsonObject));} catch (Exception e) {LOGGER.error("数据包装器执行出错....{}", e);}}
    }
  5. 使用,浏览器访问:localhost:8080/never/getBaseResult 看效果

    {
    data: {
    imgUrl: "htt://img.baidu.com/images/11.jpg",
    name: "itar"
    },
    status: 0
    }

四、细节

  1. 可能大家注意到了,我这里Filter都是有URLPattern的,都需要配置@WebFilter注解的,Spring Boot中这个注解要生效必须在启动类中写上 @ServletComponentScan ,单纯Spring 应用需要在Web.xml中配置filter。
  2. 第二种方式中response.resetBuffer();这句话非常重要,不然会出现FixedLengthOverflowException , 主要作用就是将buffer里面的数据清空
  3. 原理说白了就是提前通过你的Filter将ServletOutputStream输出到页面上去,只要执行了write方法,页面就有数据了,大家可以执行测试

五、代码下载

​ 点击这里 注意切换到 duplicate_submissions 这个分支

END ,希望有所帮助

我的博客原文,希望大家也支持一下哈 ^_^ 程序员的天花板 ,排版可能好看一些

这篇关于利用HttpServletResponseWrapper对reponse进行包装的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1039311

相关文章

Nginx中配置使用非默认80端口进行服务的完整指南

《Nginx中配置使用非默认80端口进行服务的完整指南》在实际生产环境中,我们经常需要将Nginx配置在其他端口上运行,本文将详细介绍如何在Nginx中配置使用非默认端口进行服务,希望对大家有所帮助... 目录一、为什么需要使用非默认端口二、配置Nginx使用非默认端口的基本方法2.1 修改listen指令

MySQL按时间维度对亿级数据表进行平滑分表

《MySQL按时间维度对亿级数据表进行平滑分表》本文将以一个真实的4亿数据表分表案例为基础,详细介绍如何在不影响线上业务的情况下,完成按时间维度分表的完整过程,感兴趣的小伙伴可以了解一下... 目录引言一、为什么我们需要分表1.1 单表数据量过大的问题1.2 分表方案选型二、分表前的准备工作2.1 数据评估

MySQL进行分片合并的实现步骤

《MySQL进行分片合并的实现步骤》分片合并是指在分布式数据库系统中,将不同分片上的查询结果进行整合,以获得完整的查询结果,下面就来具体介绍一下,感兴趣的可以了解一下... 目录环境准备项目依赖数据源配置分片上下文分片查询和合并代码实现1. 查询单条记录2. 跨分片查询和合并测试结论分片合并(Shardin

SpringBoot结合Knife4j进行API分组授权管理配置详解

《SpringBoot结合Knife4j进行API分组授权管理配置详解》在现代的微服务架构中,API文档和授权管理是不可或缺的一部分,本文将介绍如何在SpringBoot应用中集成Knife4j,并进... 目录环境准备配置 Swagger配置 Swagger OpenAPI自定义 Swagger UI 底

基于Python Playwright进行前端性能测试的脚本实现

《基于PythonPlaywright进行前端性能测试的脚本实现》在当今Web应用开发中,性能优化是提升用户体验的关键因素之一,本文将介绍如何使用Playwright构建一个自动化性能测试工具,希望... 目录引言工具概述整体架构核心实现解析1. 浏览器初始化2. 性能数据收集3. 资源分析4. 关键性能指

Nginx进行平滑升级的实战指南(不中断服务版本更新)

《Nginx进行平滑升级的实战指南(不中断服务版本更新)》Nginx的平滑升级(也称为热升级)是一种在不停止服务的情况下更新Nginx版本或添加模块的方法,这种升级方式确保了服务的高可用性,避免了因升... 目录一.下载并编译新版Nginx1.下载解压2.编译二.替换可执行文件,并平滑升级1.替换可执行文件

Python进行JSON和Excel文件转换处理指南

《Python进行JSON和Excel文件转换处理指南》在数据交换与系统集成中,JSON与Excel是两种极为常见的数据格式,本文将介绍如何使用Python实现将JSON转换为格式化的Excel文件,... 目录将 jsON 导入为格式化 Excel将 Excel 导出为结构化 JSON处理嵌套 JSON:

一文解密Python进行监控进程的黑科技

《一文解密Python进行监控进程的黑科技》在计算机系统管理和应用性能优化中,监控进程的CPU、内存和IO使用率是非常重要的任务,下面我们就来讲讲如何Python写一个简单使用的监控进程的工具吧... 目录准备工作监控CPU使用率监控内存使用率监控IO使用率小工具代码整合在计算机系统管理和应用性能优化中,监

如何使用Lombok进行spring 注入

《如何使用Lombok进行spring注入》本文介绍如何用Lombok简化Spring注入,推荐优先使用setter注入,通过注解自动生成getter/setter及构造器,减少冗余代码,提升开发效... Lombok为了开发环境简化代码,好处不用多说。spring 注入方式为2种,构造器注入和setter

MySQL进行数据库审计的详细步骤和示例代码

《MySQL进行数据库审计的详细步骤和示例代码》数据库审计通过触发器、内置功能及第三方工具记录和监控数据库活动,确保安全、完整与合规,Java代码实现自动化日志记录,整合分析系统提升监控效率,本文给大... 目录一、数据库审计的基本概念二、使用触发器进行数据库审计1. 创建审计表2. 创建触发器三、Java