通过自定义filter,RequestWrapper,ResponseWrapper 处理请求和响应数据,比如修改请求体和响应体的字符编码

1.request 和 response 中的数据都是 存在流中的(缓存中)获取一次就没有了,需要重新写回去。所以需要两个包装类分别继承HttpServletRequestWrapper 和 HttpServletResponseWrapper 对 request和response进行包装,从中获取数据。

package com.hoau.monitor.servlet;import javax.servlet.ReadListener;
import javax.servlet.ServletInputStream;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletRequestWrapper;
import java.io.*;public class RequestWrapper extends HttpServletRequestWrapper {private final String body;public RequestWrapper(HttpServletRequest request) throws IOException {super(request);StringBuilder stringBuilder = new StringBuilder();BufferedReader bufferedReader = null;try {InputStream inputStream = request.getInputStream();if (inputStream != null) {bufferedReader = new BufferedReader(new InputStreamReader(inputStream));char[] charBuffer = new char[128];int bytesRead = -1;while ((bytesRead = bufferedReader.read(charBuffer)) > 0) {stringBuilder.append(charBuffer, 0, bytesRead);}} else {stringBuilder.append("");}} catch (IOException ex) {throw ex;} finally {if (bufferedReader != null) {try {bufferedReader.close();} catch (IOException ex) {throw ex;}}}body = stringBuilder.toString();}@Overridepublic ServletInputStream getInputStream() throws IOException {final ByteArrayInputStream byteArrayInputStream = new ByteArrayInputStream(body.getBytes());ServletInputStream servletInputStream = new ServletInputStream() {public boolean isFinished() {return false;}public boolean isReady() {return false;}public void setReadListener(ReadListener readListener) {}public int read() throws IOException {return byteArrayInputStream.read();}};return servletInputStream;}@Overridepublic BufferedReader getReader() throws IOException {return new BufferedReader(new InputStreamReader(this.getInputStream()));}public String getBody() {return this.body;}}
package com.hoau.monitor.servlet;import javax.servlet.ServletOutputStream;
import javax.servlet.WriteListener;
import javax.servlet.http.HttpServletResponse;
import javax.servlet.http.HttpServletResponseWrapper;
import java.io.*;/*** @param* @author * @create 2019-06-10 13:31* @return* @description :*/
public class ResponseWrapper extends HttpServletResponseWrapper {private ByteArrayOutputStream bytes = new ByteArrayOutputStream();private HttpServletResponse response;private PrintWriter pwrite;public ResponseWrapper(HttpServletResponse response) {super(response);this.response = response;}@Overridepublic ServletOutputStream getOutputStream() throws IOException {// 将数据写到 byte 中return new MyServletOutputStream(bytes);}/*** 重写父类的 getWriter() 方法,将响应数据缓存在 PrintWriter 中*/@Overridepublic PrintWriter getWriter() throws IOException {try {pwrite = new PrintWriter(new OutputStreamWriter(bytes, "utf-8"));} catch (UnsupportedEncodingException e) {e.printStackTrace();}return pwrite;}/*** 获取缓存在 PrintWriter 中的响应数据** @return*/public byte[] getBytes() {if (null != pwrite) {pwrite.close();return bytes.toByteArray();}if (null != bytes) {try {bytes.flush();} catch (IOException e) {e.printStackTrace();}}return bytes.toByteArray();}class MyServletOutputStream extends ServletOutputStream {private ByteArrayOutputStream ostream;public MyServletOutputStream(ByteArrayOutputStream ostream) {this.ostream = ostream;}@Overridepublic void write(int b) throws IOException {ostream.write(b); // 将数据写到 stream 中}@Overridepublic boolean isReady() {return false;}@Overridepublic void setWriteListener(WriteListener listener) {}}}

自定义Filter,主要重写doFilter方法,在doFilter方法中通过包装类RequestWrapper,ResponseWrapper可以获取或修改请求和响应体内容

package com.hoau.monitor.filter;import com.alibaba.dubbo.config.annotation.Reference;
import com.alibaba.fastjson.JSONObject;
import com.hoau.common.exception.PSBCommonException;
import com.hoau.monitor.entity.MonitorData;
import com.hoau.monitor.facade.IhoauMonitorDataFacade;
import com.hoau.monitor.servlet.RequestWrapper;
import com.hoau.monitor.servlet.ResponseWrapper;
import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;
import org.springframework.beans.factory.annotation.Qualifier;
import org.springframework.stereotype.Component;
import org.springframework.web.servlet.HandlerMapping;import javax.servlet.*;
import javax.servlet.annotation.WebFilter;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;@Component
@WebFilter(urlPatterns = {"/psb/*"}, filterName = "monitorDataFilter")
public class MonitorDataFilter implements Filter {@Reference(group = "HoauMonitorFacadeServiceImpl")@Qualifier("ihoauMonitorDataServiceImpl")IhoauMonitorDataFacade ihoauMonitorDataServiceImpl;Log log = LogFactory.getLog(MonitorDataFilter.class);private MonitorData monitorData;@Overridepublic void init(FilterConfig filterConfig) throws ServletException {}@Overridepublic void doFilter(ServletRequest servletRequest, ServletResponse servletResponse, FilterChain filterChain) throws IOException, ServletException {//此处可设置http请求和响应的字符编码格式servletRequest.setCharacterEncoding("gbk");servletResponse.setCharacterEncoding("gbk");HttpServletRequest httpServletRequest = (HttpServletRequest) servletRequest;HttpServletResponse httpServletResponse = (HttpServletResponse) servletResponse;RequestWrapper requestWrapper = new RequestWrapper(httpServletRequest);ResponseWrapper responseWrapper = new ResponseWrapper(httpServletResponse);filterChain.doFilter(requestWrapper, responseWrapper);monitorData = new MonitorData();String params = null;String paramGet = null;String paramVar = null;String val = null;
//        String meth = httpServletRequest.getMethod();// 获取参数的方式实际跟接收方式相关(@RequestBody @RequestParam @PathVariable),多种注解一块用时参数拼接?// 与请求方式无直接关系(post get put delete)params = requestWrapper.getBody();paramGet = JSONObject.toJSONString(httpServletRequest.getParameterMap());paramVar = JSONObject.toJSONString(httpServletRequest.getAttribute(HandlerMapping.URI_TEMPLATE_VARIABLES_ATTRIBUTE));if (!"{}".equals(paramGet)&& !"null".equals(paramGet)) {params = params + paramGet;}if (!"{}".equals(paramVar) && !"null".equals(paramVar)) {params = params + paramVar;}log.info("========1.请求参数:" + params);monitorData.setInterAddress(httpServletRequest.getRequestURL().toString());monitorData.setAccAccount("");monitorData.setAccIP(httpServletRequest.getRemoteAddr());monitorData.setAccParams(params);monitorData.setInterType("");byte[] bytes = responseWrapper.getBytes();val = new String(bytes, "UTF-8");monitorData.setRetVal(JSONObject.toJSONString(val));log.info("========2.返回值:" + val);//将数据 再写到 response 中byte[] gbks = val.getBytes("gbk");//这里编码转换操作,这里转换的一定要与接口响应的编码相同servletResponse.setContentLength(gbks.length);//这一步一定要有,否则字节数组长度不够会造成线程一直等待而阻塞servletResponse.getOutputStream().write(gbks);servletResponse.getOutputStream().flush();servletResponse.getOutputStream().close();try {ihoauMonitorDataServiceImpl.addMonitorData(monitorData);} catch (PSBCommonException e) {e.printStackTrace();}}@Overridepublic void destroy() {}
}

除了filter 也可以通过切面(配合注解)获取和修改 ServletRequest , ServletResponse ,及请求和响应数据

package com.hoau.common.auth;import java.lang.annotation.*;/*** @author : * @create : 2019-07-25 10:24* @description :监控注解 应用此注解时,请求的第一个参数必须是 HttpServletRequest*/
@Documented
@Target({ElementType.METHOD})
@Retention(RetentionPolicy.RUNTIME)
public @interface HoauMonitorAnnotation {}
package com.hoau.common.auth;import com.alibaba.fastjson.JSONObject;
import com.hoau.common.entity.MonitorData;
import com.hoau.common.servlet.RequestWrapper;
import com.hoau.common.servlet.ResponseWrapper;
import com.hoau.common.util.HttpUtils;
import com.hoau.common.util.JsonUtils;
import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.Around;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Pointcut;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.core.annotation.Order;
import org.springframework.stereotype.Component;
import org.springframework.web.context.request.RequestContextHolder;
import org.springframework.web.context.request.ServletRequestAttributes;
import org.springframework.web.servlet.HandlerMapping;import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;
import java.time.Instant;
import java.time.OffsetDateTime;
import java.time.ZoneOffset;
import java.util.HashMap;
import java.util.Map;/*** @author : xilipeng* @create : 2019-07-25 10:25* @description :HoauMonitorAnnotation 监控切面*/
@Aspect
@Order(0)//监控模块优先级必须高于 auth 校验模块,否则安全校验不通过时会是监控通知不执行
@Component
public class MonitorAspect {@Value("${monitor.url}")protected String monitorUrl;@Pointcut("execution(public * com.hoau.*.controller.*.*(..)) && "+ "@annotation(com.hoau.common.auth.HoauMonitorAnnotation)")public void monitorPointcut() {}@Around("monitorPointcut()")public Object monitorAdvice(ProceedingJoinPoint pjp) throws IOException {System.out.println("========> system into MonitorAspect ");Object result = null;Object[] args = pjp.getArgs();
//        BackObject backObject = new BackObject();MonitorData monitorData = new MonitorData();try {//获取请求中的 HttpServletRequestServletRequestAttributes requestAttributes = (ServletRequestAttributes) RequestContextHolder.getRequestAttributes();HttpServletRequest httpServletRequest = requestAttributes.getRequest();HttpServletResponse response = requestAttributes.getResponse();
//            PrintWriter writer = response.getWriter();
//            HttpEncodingAutoConfiguration httpEncodingAutoConfiguration;
//            ServletOutputStream outputStream = response.getOutputStream();
//            outputStream.ResponseWrapper responseWrapper = new ResponseWrapper(response);byte[] bytes =responseWrapper.getBytes();if (null != httpServletRequest) {//获取请求参数RequestWrapper requestWrapper = new RequestWrapper(httpServletRequest);String params = requestWrapper.getBody();String paramGet = JSONObject.toJSONString(httpServletRequest.getParameterMap());if (!"{}".equals(paramGet) && !"null".equals(paramGet)) {params += paramGet;}//获取接口路径String uri = httpServletRequest.getRequestURI().toString();
//              String meth = httpServletRequest.getMethod();// 获取参数的方式实际跟接收方式相关(@RequestBody @RequestParam @PathVariable),多种注解一块用时参数拼接?// 与请求方式无直接关系(post get put delete)Map<String, Object> pathVariables = (HashMap<String, Object>) httpServletRequest.getAttribute(HandlerMapping.URI_TEMPLATE_VARIABLES_ATTRIBUTE);if (null != pathVariables && !pathVariables.isEmpty()) {String paramVar = JSONObject.toJSONString(pathVariables);params += paramVar;for (String key : pathVariables.keySet()) {String val = "/" + pathVariables.get(key).toString();uri = uri.replace(val, "");}}monitorData.setInterAddress(uri);System.out.println("========1.接口地址:" + uri);monitorData.setAccAccount("");monitorData.setAccIP(httpServletRequest.getRemoteAddr());//存入 ES 的时间OffsetDateTime offsetDateTime = Instant.now().atOffset(ZoneOffset.ofHours(8));//2019-08-22T11:49:53.054+08:00monitorData.setAccTime(offsetDateTime.toString());System.out.println("========Instant.now().atOffset(ZoneOffset.ofHours(8)):" + offsetDateTime.toString());monitorData.setAccParams(params);System.out.println("========2.请求参数:" + params);monitorData.setInterType("");} else if (null != args) {//如果未获取到 HttpServletRequest 就只保存请求参数和结果 和请求时间//存入 ES 的时间OffsetDateTime offsetDateTime = Instant.now().atOffset(ZoneOffset.ofHours(8));//2019-08-22T11:49:53.054+08:00monitorData.setAccTime(offsetDateTime.toString());monitorData.setAccParams(JsonUtils.toJson(args));System.out.println("========1.接口地址:请求参数中未获取到  httpServletRequest");System.out.println("========2.请求参数:" + JsonUtils.toJson(args));}result = pjp.proceed();} catch (Throwable e) {e.printStackTrace();
//            backObject.setBackStatusCode(RestErrorCodeConstants.STATUS_SYSTEM_ERROR);
//            backObject.setBackStatusStr(RestErrorCodeConstants.STATUS_SYSTEM_ERROR_INFO + e.getMessage());
//            result = JsonUtils.toJson(backObject);}monitorData.setRetVal(JSONObject.toJSONString(result));System.out.println("========3.返回值:" + JSONObject.toJSONString(result));try {HttpUtils.sendPostRequest(monitorUrl, JsonUtils.toJson(monitorData), "form");} catch (Exception e) {System.out.println("========监控添加失败:" + e.getMessage());e.printStackTrace();}System.out.println("========> system out MonitorAspect ");return result;}
}

使用切面也需要filter

package com.hoau.common.filter;import com.hoau.common.servlet.RequestWrapper;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.stereotype.Component;import javax.servlet.*;
import javax.servlet.annotation.WebFilter;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;/*** @author : * @create : 2019-09-01 16:23* @description :*/
@Component//在 common 包中需要加上此注解,否则扫描不到
@WebFilter(urlPatterns = {"/*"}, filterName = "requestFilter")
public class RequestFilter implements Filter {private static Logger log = LoggerFactory.getLogger(RequestFilter.class);@Overridepublic void destroy() {}@Overridepublic void doFilter(ServletRequest request, ServletResponse response,FilterChain filterChain) throws IOException, ServletException {HttpServletRequest httpServletRequest = (HttpServletRequest) request;HttpServletResponse httpServletResponse = (HttpServletResponse) response;RequestWrapper requestWrapper = new RequestWrapper(httpServletRequest);
//        ResponseWrapper responseWrapper = new ResponseWrapper(httpServletResponse);filterChain.doFilter(requestWrapper, httpServletResponse);}@Overridepublic void init(FilterConfig arg0) throws ServletException {}}

Spring/SpringBoot 过滤器修改、获取http 请求request中的参数 和 response返回值,比如修改请求体和响应体的字符编码相关推荐

  1. java过滤器修改响应,在过滤器中实现修改http请求体和响应体

    在一些业务场景中,需要对http的请求体和响应体做加解密的操作,如果在controller中来调用加解密函数,会增加代码的耦合度,同时也会增加调试的难度. 参考spring中http请求的链路,选择过 ...

  2. django视图(request请求response返回值)

    目录 一.视图函数介绍 二.HTTPRequest对象(get请求与post请求) 2.1  url路径参数(前端向后端传递参数) 2.2  get 请求传参 2.3.1  url ?后面的key v ...

  3. Spring MVC使用篇(八)—— 处理器(Handler)方法的返回值

    文章目录 1.演示项目环境搭建 1.1 演示项目工程结构 1.2 演示项目依赖的基础jar包 1.3 配置web.xml 1.4 配置Spring MVC核心配置文件 2.返回ModelAndView ...

  4. Java中request有哪些方法,Java--获取request中所有参数的方法

    java获取request中的参数.java解析URL问号后的参数.有时候我们需要从request中获取参数,或者获取拼接在Url后面的参数,有时候一个一个去拿有点麻烦,一起拿出来放在一个map里面需 ...

  5. springboot自定义参数处理器和返回值处理器

    参数处理器(ArgumentResolvers)和返回参数处理器(ReturnValueHandlers) 在我们调用controller层组件时,Springboot实际上是使用代理模式进行调用,s ...

  6. vert.x java post请求无法接收到post请求body中的参数

    最近才了解到这个框架vert.x,这个框架在我看来是很不错的. 1.首先基于netty,这个优点就不说了. 2.异步非阻塞的网络框架. 3.轻量级框架,启动真的是超级快,如果你用过spring的框架, ...

  7. Java--获取request中所有参数的方法

    我们通常用到request获取某个参数的方法: String value=request.getparameter("key"); 如果想要获取request中所有的参数呢? re ...

  8. linux修改key文件夹,linux 中如何根据xml文件的key来修改value呢?

    源自:4-4 Linux中 grep命令 linux 中如何根据xml文件的key来修改value呢? 有个xml文件(server.xml)比如: .... 请问该如何获取加粗的   style=& ...

  9. 小程序通过扫码进入时获取二维码中的参数(采坑)

    情景:假如你想在通过二维码进入小程序时候,根据二维码中参数进行一些别的操作 1 开发阶段,在开发者工具中 微信开发者工具中编译条件中有一个自定义编译模式,同时可以携带参数,启动参数中写你要携带的参数( ...

最新文章

  1. CopyCat 代码克隆检测发布,剑指开源软件抄袭
  2. Orcale11g单机安装与卸载
  3. 斯特芬森迭代法(Python)
  4. 【LeetCode从零单排】No19.RemoveNthNodeFromEndofList
  5. Java中,String类型和包装类型作为参数传递时,是属于值传递还是引用传递呢?...
  6. react配合python_部署React前端和Django后端的3种方法
  7. 第 4-1 课:BIO、NIO、AIO 详解 + 面试题
  8. 冬天你的车热的正确吗?如何正确热车?
  9. FortiGate设备管理
  10. [转载]Git安装以及使用Git 管理个人文档
  11. 数据可视化平台-智慧安防可视化管理系统-解决楼宇监控管理难题
  12. win10系统点击控制台自定义快捷键失效问题解决
  13. c语言头文件有哪些intr,有没有大神帮帮忙
  14. 计算机分子模拟聚乙烯,高分子物理实验思考题@中科大.pdf
  15. 1.Redis客户端
  16. 会员付费超前点播模式争议背后,我们该怎么看待在线视频的未来?
  17. 计算机会计和传统手工会计的区别,简要比较手工会计和计算机数据处理方式的区别...
  18. mysql查看数据库的容量及表容量
  19. css中如何设置hr的样式?css hr标签多种样式(图文)
  20. 分享:无盘工作站的系统构架指南(转)

热门文章

  1. Android 动态分区详解(六) 动态分区的底层机制
  2. FME进阶视频教程: FME使用技巧之高级扇出,讲解在FME中输出数据的高级方式,满足数据个性化分类输出的需求
  3. 用C语言编写简单的通讯录(大一课程作业)
  4. 黑客攻陷'360安全论坛' 偷窃网游和QQ帐号
  5. 红黑树从头至尾插入和删除结点的全程演示图
  6. 1093. 大样本统计-正常统计
  7. 【游戏】任天堂Nintendo Switch超详细购买/使用指南以及注意事项(根据自己使用持续更新中...)
  8. 推流(RTSP/RTMP)
  9. LeetCode881:救生艇 (C、C++实现)
  10. Servlet获取Excel中数据的两种方式