Filter自定义过滤器

本文介绍了一个Spring MVC项目中Filter过滤器的实现案例,通过配置FilterRegistrationBean和自定义ParamterFilter,实现对特定URL模式的请求拦截,特别关注POST请求的数据读取与日志记录。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

最近接触到filter过滤器,之前都没有接触到,这次在组长大人得帮助下对filter过滤器有了一点概念,组长大人还给了一个例子,记录一下。

@Configuration
public class SpringMvcConfig implements WebMvcConfigurer {
    @Bean
    public FilterRegistrationBean filterRegist() {
        FilterRegistrationBean frBean = new FilterRegistrationBean();
        //定义一个 ParamterFilter过滤器
        frBean.setFilter(new ParamterFilter());
        //需要拦截的地址
        frBean.addUrlPatterns("/cxf/*");
        //设置注册名称,若没有指定,使用bean名称
        frBean.setName("WebAsscessFilterMVC");
        return frBean;
    }
}
@Component
public class ParamterFilter extends OncePerRequestFilter {
    @Override
    protected void doFilterInternal(HttpServletRequest request, HttpServletResponse response, FilterChain filterChain) throws ServletException, IOException {
        if(HttpMethod.POST.equals(HttpMethod.resolve(request.getMethod()))){
        //调用封装好的一个Wrapper
            PostRequestWrapper postRequestWrapper = new PostRequestWrapper(request);
            wirteLog(postRequestWrapper,response);
            filterChain.doFilter(postRequestWrapper,response);
        }else {
            wirteLog(request,response);
            filterChain.doFilter(request,response);
        }
    }
    protected void wirteLog(HttpServletRequest request, HttpServletResponse response) throws IOException {
        InputStream inputStream = request.getInputStream();
        ByteArrayOutputStream result = new ByteArrayOutputStream();
        byte[] buf = new byte[2048];
        int len = 0;
        while((len=inputStream.read(buf))>0){
            result.write(buf,0,len);
        }
        logger.debug(result.toString());
    }
}
public class PostRequestWrapper extends HttpServletRequestWrapper {
    private Logger logger = LoggerFactory.getLogger(PostRequestWrapper.class);
    private final byte[] bodyData;
    /**
     * Constructs a request object wrapping the given request.
     *
     * @param request The request to wrap
     * @throws IllegalArgumentException if the request is null
     */
    public PostRequestWrapper(HttpServletRequest request) {
        super(request);
        bodyData = getByteFromRequest(request);
    }

    /**
     * 获取Body中的字符中表示
     * @return
     */
    public String getBodyString() {
        if(bodyData != null && bodyData.length>0){
            return new String(bodyData);
        }
        return null;
    }

    /**
     *从Body体中获取数据并保存
     * @param request
     * @return
     */
    protected byte[] getByteFromRequest(HttpServletRequest request){
        ByteArrayOutputStream result = new ByteArrayOutputStream();
        try {
            //BufferedReader reader = new BufferedReader(new InputStreamReader(request.getInputStream()));
            byte[] buf = new byte[2048];
            int len = 0;
            while ((len=request.getInputStream().read(buf))>0){
                result.write(buf,0,len);
            }
        } catch (IOException e) {
            logger.error("{}",e);
        }
        return result.toByteArray();
    }
    @Override
    public BufferedReader getReader() throws IOException {
        return new BufferedReader(new InputStreamReader(getInputStream()));
    }

    @Override
    public ServletInputStream getInputStream() throws IOException {

        final ByteArrayInputStream inputStream = new ByteArrayInputStream(bodyData);

        return new ServletInputStream() {
            @Override
            public int read() throws IOException {
                return inputStream.read();
            }
            @Override
            public boolean isFinished() {
                return false;
            }
            @Override
            public boolean isReady() {
                return false;
            }
            @Override
            public void setReadListener(ReadListener readListener) {
            }
        };
    }
}

这样就ok啦,就能在日志中查看到拦截信息。
知识嘛,慢慢积累,就好啦。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值