个人学习系列 - 解决拦截器操作请求参数后台无法获取

由于项目需要使用拦截器对请求参数进行操作,可是请求流只能操作一次,导致后面方法不能再获取流了。

新建SpringBoot项目

1. 新建拦截器WebConfig.java
/**
 * @date: 2023/2/6 11:21
 * @author: zhouzhaodong
 * @description:
 */
@Configuration
public class WebConfig implements WebMvcConfigurer {
    /**
     * 添加Web项目的拦截器
     */
    @Override
    public void addInterceptors(InterceptorRegistry registry) {
        // 对所有访问路径,都通过MyInterceptor类型的拦截器进行拦截
        // 放行登录页,登陆操作,静态资源
        registry.addInterceptor(new MyInterceptor()).addPathPatterns("/**")
                .excludePathPatterns("/", "/login", "/index.html", "/user/login", "/css/**", "/images/**", "/js/**", "/fonts/**");
    }
}
2. 获取请求参数并处理逻辑
/**
 * @date: 2023/2/6 12:46
 * @author: zhouzhaodong
 * @description: 获取请求参数并处理
 */
public class RequestWrapper extends HttpServletRequestWrapper {
    private final Logger logger = LoggerFactory.getLogger(RequestWrapper.class);
    private final byte[] body;

    public RequestWrapper(HttpServletRequest request) {
        super(request);
        String sessionStream = getBodyString(request);
        body = sessionStream.getBytes(StandardCharsets.UTF_8);
    }

    public String getBodyString() {
        return new String(body, StandardCharsets.UTF_8);
    }

    /**
     * @date: 2023/2/6 12:46
     * @author: zhouzhaodong
     * @description: 获取请求Body
     */
    public String getBodyString(final ServletRequest request) {
        StringBuilder sb = new StringBuilder();
        InputStream inputStream = null;
        BufferedReader reader = null;
        try {
            inputStream = cloneInputStream(request.getInputStream());
            reader = new BufferedReader(new InputStreamReader(inputStream, StandardCharsets.UTF_8));
            String line = "";
            while ((line = reader.readLine()) != null) {
                sb.append(line);
            }
        } catch (IOException e) {
            e.printStackTrace();
        } finally {
            if (inputStream != null) {
                try {
                    inputStream.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
            if (reader != null) {
                try {
                    reader.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
        }
        logger.info("获取body请求参数:" + sb);
        return sb.toString();
    }

    /**
     * @date: 2023/2/6 12:46
     * @author: zhouzhaodong
     * @description: 复制输入流
     */
    public InputStream cloneInputStream(ServletInputStream inputStream) {
        ByteArrayOutputStream byteArrayOutputStream = new ByteArrayOutputStream();
        byte[] buffer = new byte[1024];
        int len;
        try {
            while ((len = inputStream.read(buffer)) > -1) {
                byteArrayOutputStream.write(buffer, 0, len);
            }
            byteArrayOutputStream.flush();
        } catch (IOException e) {
            e.printStackTrace();
        }
        return new ByteArrayInputStream(byteArrayOutputStream.toByteArray());
    }

    @Override
    public BufferedReader getReader() {
        return new BufferedReader(new InputStreamReader(getInputStream()));
    }

    @Override
    public ServletInputStream getInputStream() {

        final ByteArrayInputStream bais = new ByteArrayInputStream(body);
        return new ServletInputStream() {

            @Override
            public int read() {
                return bais.read();
            }

            @Override
            public boolean isFinished() {
                return false;
            }

            @Override
            public boolean isReady() {
                return false;
            }

            @Override
            public void setReadListener(ReadListener readListener) {

            }
        };
    }
}
3. 实现HandlerInterceptor接口
/**
 * @date: 2023/2/6 11:19
 * @author: zhouzhaodong
 * @description: 实现HandlerInterceptor接口
 */

public class MyInterceptor implements HandlerInterceptor {

    private final Logger logger = LoggerFactory.getLogger(MyInterceptor.class);

    /**
     * @date: 2023/2/6 11:19
     * @author: zhouzhaodong
     * @description: 访问控制器方法前执行
     */
    @Override
    public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) {
        logger.info("================进入拦截器======================");
        logger.info(new Date() + "--preHandle:" + request.getRequestURL());
        logger.info("***************************【RequestBeginning】***************************");
        logger.info("----------------StartProcessingRequest----------------");
        try {
            long currentTime = System.currentTimeMillis();
            SimpleDateFormat formatter = new SimpleDateFormat("yyyy/MM/dd HH:mm:ss");
            Date date = new Date(currentTime);
            logger.info("CurrentTime: {}", formatter.format(date));
            logger.info("ResponseTime: {}", (System.currentTimeMillis() - currentTime) + "ms");
            String requestUrl = request.getRequestURI();
            logger.info("RequestURL: {} ", requestUrl);
            logger.info("GetMethod: {}", handler);
            String method = request.getMethod();
            logger.info("Method: {}", method);
            //获取请求参数
            RequestWrapper requestWrapper = new RequestWrapper(request);
            //这里getBodyString()方法无参数
            logger.info("RequestBody: {}", requestWrapper.getBodyString());
        } catch (Exception e) {
            logger.error("MVC业务处理-拦截器异常:", e);
        }
        logger.info("-------------------------End-------------------------");
        return true;
    }

    /**
     * @date: 2023/2/6 12:46
     * @author: zhouzhaodong
     * @description: 访问控制器方法后执行
     */
    @Override
    public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView) {
        logger.info(new Date() + "--postHandle:" + request.getRequestURL());
    }

    /**
     * @date: 2023/2/6 12:46
     * @author: zhouzhaodong
     * @description: postHandle方法执行完成后执行,一般用于释放资源
     */
    @Override
    public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) {
        logger.info(new Date() + "--afterCompletion:" + request.getRequestURL());
    }
}
4. application.yml
spring:
  main:
    # 当出现相同名字的类进行注册时,准许覆盖注册
    allow-bean-definition-overriding: true
5. 启动类添加@ServletComponentScan注解
6. 新建TestController.java
/**
 * @date: 2023/2/6 12:24
 * @author: zhouzhaodong
 * @description: 测试
 */
@RestController
public class TestController {

    @PostMapping("/one/abc")
    public String abc(@RequestBody User user){
        return user.getName();
    }
}
7. 不进行处理先测试看结果

请求结果:
在这里插入图片描述

控制台输出:
在这里插入图片描述

可以看出,流被读取了一次然后后台就获取不到了。

8. 通过过滤器获取参数然后传到后面程序中
/**
 * @date: 2023/2/6 12:47
 * @author: zhouzhaodong
 * @description:
 */
@Component
@WebFilter(urlPatterns = "/*", filterName = "channelFilter")
public class ChannelFilter implements Filter {
    private final Logger logger = LoggerFactory.getLogger(ChannelFilter.class);

    @Override
    public void init(FilterConfig filterConfig) {

    }

    @Override
    public void doFilter(ServletRequest servletRequest, ServletResponse servletResponse, FilterChain filterChain) throws IOException, ServletException {
        logger.info("================进入过滤器======================");
        // 防止流读取一次后就没有了, 所以需要将流继续写出去
        HttpServletRequest httpServletRequest = (HttpServletRequest) servletRequest;
        RequestWrapper requestWrapper = new RequestWrapper(httpServletRequest);
        filterChain.doFilter(requestWrapper, servletResponse);
    }

    @Override
    public void destroy() {

    }
}
9. 流处理后再进行测试

请求结果:
在这里插入图片描述
控制台输出:
在这里插入图片描述

解决啦!!!

源代码路径

https://github.com/zhouzhaodong/springboot/tree/master/springboot-interceptor

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
可以通过编写拦截器来实现修改/添加前端传输到后台的header和cookie参数。 首先,定义一个拦截器类,实现 HandlerInterceptor 接口,并在其中重写 preHandle 方法。在 preHandle 方法中,可以通过 request.getHeader() 和 request.getCookies() 方法获取前端传输过来的 header 和 cookie 参数,然后进行修改/添加,最后再通过 response.setHeader() 和 response.addCookie() 方法将修改后的参数传输到后台。 下面是一个简单的示例: ```java public class CustomInterceptor implements HandlerInterceptor { @Override public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception { // 获取前端传输过来的 header 参数 String token = request.getHeader("token"); // 修改/添加 header 参数 response.setHeader("token", "newToken"); // 获取前端传输过来的 cookie 参数 Cookie[] cookies = request.getCookies(); // 修改/添加 cookie 参数 Cookie cookie = new Cookie("newCookie", "cookieValue"); response.addCookie(cookie); // 返回 true 表示继续执行后续的处理器和拦截器 return true; } } ``` 然后,在 Spring Boot 项目中配置拦截器,可以通过实现 WebMvcConfigurer 接口,在 addInterceptors 方法中添加拦截器: ```java @Configuration public class InterceptorConfig implements WebMvcConfigurer { @Override public void addInterceptors(InterceptorRegistry registry) { registry.addInterceptor(new CustomInterceptor()).addPathPatterns("/**"); } } ``` 这样,当前端发送请求时,拦截器就会拦截请求,并修改/添加前端传输到后台的 header 和 cookie 参数
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值