当前位置:   article > 正文

傻傻的分不清,Spring Boot 过滤器、拦截器、监听器对比及使用场景_springcloud过滤器和拦截器

springcloud过滤器和拦截器

路人 路人甲Java 2023-12-05 08:31 发表于上海

收录于合集

#路人甲java58个

#SpringMVC系列33个

您好,我是路人,更多优质文章见个人博客:http://itsoku.com

Java充电社

Java充电社,专注分享Java技术干货,包括多线程、JVM、SpringBoot、SpringCloud、Dubbo、Zookeeper、Redis、架构设计、微服务、消息队列、Git、面试题、程序员攻略、最新动态等。

28篇原创内容

公众号

一、关系图理解

图片

二、区别

1.过滤器

  • 过滤器是在web应用启动的时候初始化一次, 在web应用停止的时候销毁

  • 可以对请求的URL进行过滤, 对敏感词过滤

  • 挡在拦截器的外层

  • 实现的是 javax.servlet.Filter 接口,是 Servlet 规范的一部分

  • 在请求进入容器后,但在进入servlet之前进行预处理,请求结束是在servlet处理完以后

  • 依赖Web容器

  • 会多次执行

1.1HttpServletRequestWrapper

在请求到达之前对 request 进行修改

  1. package com.dingwen.lir.filter;
  2. import lombok.extern.slf4j.Slf4j;
  3. import javax.servlet.http.HttpServletRequest;
  4. import javax.servlet.http.HttpServletRequestWrapper;
  5. import java.util.Arrays;
  6. /**
  7.  *  在请求到达之前对 request 进行修改
  8.  *
  9.  * @author dingwen
  10.  */
  11. @Slf4j
  12. public class RequestWrapper extends HttpServletRequestWrapper {
  13.     public RequestWrapper(HttpServletRequest request) {
  14.         super(request);
  15.         log.info("RequestWrapper");
  16.     }
  17.     @Override
  18.     public String getParameter(String name) {
  19.         // 可以对请求参数进行过滤
  20.         return super.getParameter(name);
  21.     }
  22.     @Override
  23.     public String[] getParameterValues(String name) {
  24.         // 对请求参数值进行过滤
  25. //        String[] values =super.getRequest().getParameterValues(name);
  26. //        return super.getParameterValues(name);
  27.         return "t e s t".split(" ");
  28.     }
  29. }
1.2 OncePerRequestFilter

OncePerRequestFilter,顾名思义,它能够确保在一次请求中只通过一次filter

  1. package com.dingwen.lir.filter;
  2. import lombok.extern.slf4j.Slf4j;
  3. import org.springframework.web.filter.OncePerRequestFilter;
  4. import javax.servlet.FilterChain;
  5. import javax.servlet.ServletException;
  6. import javax.servlet.http.HttpServletRequest;
  7. import javax.servlet.http.HttpServletResponse;
  8. import java.io.IOException;
  9. import java.io.PrintWriter;
  10. import java.util.Arrays;
  11. /**
  12.  * 请求过滤器
  13.  * OncePerRequestFilter:
  14.  * OncePerRequestFilter,顾名思义,它能够确保在一次请求中只通过一次filter.
  15.  * 大家常识上都认为,一次请求本来就只filter一次,为什么还要由此特别限定呢,往往我们的常识和实际的实现并不真的一样,经过一番资料的查阅,此方法是为了兼容不同的web container,
  16.  * 也就是说并不是所有的container都入我们期望的只过滤一次,servlet版本不同,执行过程也不同,
  17.  * 因此,为了兼容各种不同运行环境和版本,默认filter继承OncePerRequestFilter是一个比较稳妥的选择。
  18.  *
  19.  * @author dingwen
  20.  */
  21. @Slf4j
  22. public class RequestFilter extends OncePerRequestFilter {
  23.     @Override
  24.     public void destroy() {
  25.         super.destroy();
  26.         log.info("RequestFilter destroy");
  27.     }
  28.     /*
  29.             OncePerRequestFilter.doFilter方法中通过request.getAttribute判断当前过滤器是否已执行
  30.             若未执行过,则调用doFilterInternal方法,交由其子类实现
  31.         */
  32.     @Override
  33.     protected void doFilterInternal(HttpServletRequest httpServletRequest, HttpServletResponse httpServletResponse, FilterChain filterChain) throws ServletException, IOException {
  34.         try {
  35.             RequestWrapper requestWrapper = new RequestWrapper(httpServletRequest);
  36.             filterChain.doFilter(requestWrapper, httpServletResponse);
  37.             log.info("RequestFilter");
  38.             log.info(Arrays.toString(requestWrapper.getParameterValues("name")));
  39.         } catch (Exception exception) {
  40.             httpServletResponse.setCharacterEncoding("utf-8");
  41.             httpServletResponse.setContentType("application/json; charset=utf-8");
  42.             PrintWriter writer = httpServletResponse.getWriter();
  43.             writer.write(exception.toString());
  44.         }
  45.     }
  46. }
1.3 配置
  1. package com.dingwen.lir.configuration;
  2. import com.dingwen.lir.filter.RequestFilter;
  3. import com.dingwen.lir.filter.RequestWrapper;
  4. import org.springframework.boot.web.servlet.FilterRegistrationBean;
  5. import org.springframework.context.annotation.Bean;
  6. import org.springframework.context.annotation.Configuration;
  7. import javax.servlet.Filter;
  8. /**
  9.  * 过滤器配置类
  10.  *
  11.  * @author dingwen
  12.  */
  13. @Configuration
  14. public class FilterConfig {
  15.     @Bean
  16.     public RequestFilter requestFilter(){
  17.         return new RequestFilter();
  18.     }
  19.     @Bean
  20.     public FilterRegistrationBean<RequestFilterregistrationBean() {
  21.         FilterRegistrationBean<RequestFilter> registrationBean = new FilterRegistrationBean<>();
  22.         registrationBean.setFilter(requestFilter());
  23.         registrationBean.addUrlPatterns("/filter/*");
  24.         registrationBean.setName("RequestFilter");
  25.         //过滤器的级别,值越小级别越高越先执行
  26.         registrationBean.setOrder(1);
  27.         return registrationBean;
  28.     }
  29. }

2.拦截器

  • 实现 org.springframework.web.servlet.HandlerInterceptor 接口,动态代理

  • 拦截器应用场景, 性能分析, 权限检查, 日志记录

  • 是一个Spring组件,并由Spring容器管理,并不

  • 不依赖Tomcat等容器,是可以单独使用的。不仅能应用在web程序中,也可以用于Application、Swing等程序中

  • 是在请求进入servlet后,在进入Controller之前进行预处理的,Controller 中渲染了对应的视图之后请求结束

2.1登录拦截
  1. package com.dingwen.lir.interceptor;
  2. import com.dingwen.lir.entity.User;
  3. import org.springframework.stereotype.Component;
  4. import org.springframework.util.ObjectUtils;
  5. import org.springframework.web.servlet.HandlerInterceptor;
  6. import javax.servlet.http.HttpServletRequest;
  7. import javax.servlet.http.HttpServletResponse;
  8. import java.io.IOException;
  9. /**
  10.  * 登录拦截
  11.  *
  12.  * @author dingwen
  13.  */
  14. @Component
  15. public class PageInterceptor implements HandlerInterceptor {
  16.     @Override
  17.     public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
  18.         User user = (User)request.getSession().getAttribute("user");
  19.         if (!ObjectUtils.isEmpty(user)) {
  20.             return true;
  21.         } else {
  22.             // 不管是转发还是重定向,必须返回false。否则出现多次提交响应的错误
  23.             redirect(request, response);
  24.             return false;
  25.         }
  26.     }
  27.     /*
  28.      * 对于请求是ajax请求重定向问题的处理方法
  29.      * @param request
  30.      * @param response
  31.      *
  32.      */
  33.     public void redirect(HttpServletRequest request, HttpServletResponse response) throws IOException {
  34.         if("XMLHttpRequest".equals(request.getHeader("X-Requested-With"))){// ajax
  35.             //获取当前请求的路径
  36.             response.setHeader("Access-Control-Expose-Headers""REDIRECT,CONTENT_PATH");
  37.             //告诉ajax我是重定向
  38.             response.setHeader("REDIRECT""REDIRECT");
  39.             //告诉ajax我重定向的路径
  40.             StringBuffer url = request.getRequestURL();
  41.             String contextPath = request.getContextPath();
  42.             response.setHeader("CONTENT_PATH", url.replace(url.indexOf(contextPath) + contextPath.length(), url.length(), "/").toString());
  43.         }else{// http
  44.             response.sendRedirect( "/page/login");
  45.         }
  46.         response.getWriter().write(403);
  47.         response.setStatus(HttpServletResponse.SC_FORBIDDEN);
  48.     }
  49. }
2.2配置
  1. package com.dingwen.lir.configuration;
  2. import com.dingwen.lir.interceptor.PageInterceptor;
  3. import org.springframework.context.annotation.Configuration;
  4. import org.springframework.web.servlet.config.annotation.InterceptorRegistry;
  5. import org.springframework.web.servlet.config.annotation.ResourceHandlerRegistry;
  6. import org.springframework.web.servlet.config.annotation.ViewControllerRegistry;
  7. import org.springframework.web.servlet.config.annotation.WebMvcConfigurer;
  8. /**
  9.  * mvc 控制器配置
  10.  * MyWebMvcConfigurer: Springboot2.x以后版本使用
  11.  *
  12.  * @author dingwen
  13.  */
  14. @Configuration
  15. public class MyWebMvcConfigurer implements WebMvcConfigurer {
  16.     /*
  17.      * 拦截器依赖于Spring容器,此处拦截了所有,需要对静态资源进行放行
  18.      */
  19.     @Override
  20.     public void addInterceptors(InterceptorRegistry registry) {
  21.         // 拦截器默认的执行顺序,就是它的注册顺序,也可以通过Order手动设置控制,值越小越先执行。
  22. //        registry.addInterceptor(new PageInterceptor()).addPathPatterns("/**").order()
  23.         registry.addInterceptor(new PageInterceptor()).addPathPatterns("/**")
  24.                 .excludePathPatterns("/page/login""/user/login","/page/ajax","/static/**");
  25.     }
  26.     /*
  27.      * 不要要写控制器即可完成页面跳转访问
  28.      * @param registry
  29.      */
  30.     @Override
  31.     public void addViewControllers(ViewControllerRegistry registry) {
  32.         registry.addViewController("/page/ajax").setViewName("ajax");
  33.     }
  34.     /*
  35.      * 自定义静态资源映射
  36.         Spring Boot 默认为我们提供了静态资源映射:
  37.                 classpath:/META-INF/resources
  38.                 classpath:/resources
  39.                 classpath:/static
  40.                 classpath:/public
  41.               优先级:META-INF/resources > resources > static > public
  42.      * @param registry
  43.      *
  44.      */
  45. //    @Override
  46. //    public void addResourceHandlers(ResourceHandlerRegistry registry) {
  47.         registry.addResourceHandler("/static/**").addResourceLocations("classpath:/static/");
  48.         registry.addResourceHandler("/static/**").addResourceLocations("file:E:/static/");
  49. //    }
  50. }

3.监听器

实现 javax.servlet.ServletRequestListenerjavax.servlet.http.HttpSessionListenerjavax.servlet.ServletContextListener 等等接口

主要用来监听对象的创建与销毁的发生, 比如 session 的创建销毁, request 的创建销毁, ServletContext 创建销毁

三、注意

1.静态资源问题

SpringBoot2.x以后版本拦截器也会拦截静态资源,在配置拦截器是需要将姿态资源放行。

  1. /*
  2.  * 拦截器依赖于Spring容器,此处拦截了所有,需要对静态资源进行放行
  3.  */
  4. @Override
  5. public void addInterceptors(InterceptorRegistry registry) {
  6.     registry.addInterceptor(new PageInterceptor()).addPathPatterns("/**")
  7.             .excludePathPatterns("/page/login""/user/login","/page/ajax","/static/**");
  8. }

SpringBoot2.x 自定义静态资源映射

  1. spring:
  2.   mvc:
  3.     static-path-pattern/static/**

默认目录

  1. classpath:/META-INF/resources
  2. classpath:/resources
  3. classpath:/static
  4. classpath:/public

优先级:META-INF/resources > resources > static > public

2.登录拦截ajax重定向

由于ajax是异步的,还在当前页面进行的局部请求。当拦截到登录请求时,即使重定向也无法生效。需采用服务端给地址由前端进行跳转。详细见登录拦截器代码。

  1. // 前端处理
  2. <!DOCTYPE html>
  3. <html lang="en">
  4. <head>
  5.     <meta charset="UTF-8">
  6.     <title>AJAX</title>
  7.     <script src="https://code.jquery.com/jquery-3.0.0.min.js"></script>
  8. </head>
  9. <body>
  10.     <button>USER</button>
  11. </body>
  12. </html>
  13. <script>
  14.     $.ajaxSetup({
  15.         complete:function(xhr,status){
  16.             //拦截器实现超时跳转到登录页面
  17.             let win = window;
  18.             // 通过xhr取得响应头
  19.             let REDIRECT = xhr.getResponseHeader("REDIRECT");
  20.             //如果响应头中包含 REDIRECT 则说明是拦截器返回的需要重定向的请求
  21.             if (REDIRECT === "REDIRECT")
  22.             {
  23.                 while (win !== win.top)
  24.                 {
  25.                     win = win.top;
  26.                 }
  27.                 win.location.href = xhr.getResponseHeader("CONTEXTPATH");
  28.             }
  29.         }
  30.     });
  31.     $("button").click(function(){
  32.         $.get("/page/user"function(result){
  33.             $("div").html(result);
  34.         });
  35.     });
  36. </script>

四、测试

代码地址:

  • https://gitee.com/dingwen-gitee/filter-interceptor-study.git

1.拦截器测试

1.1启动项目访问首页
  • http://localhost:8080/page/index

由于没有登录,直接重定向到了登录页

图片

1.2输入用户名密码完成登录,调转到用户页

图片

图片

此时在访问首页

图片

1.2 退出登录

成功退出后,访问为授权的页面也相对会被重定向到登录页

图片

1.3 ajax未授权访问测试

图片

点击访问user ,由于未登录,没有全权访问。在前端进行了页面跳转,转到了登录页。

图片

2.过滤器测试

图片

图片

可以看到过滤器进行了相对应的处理,重写的getParameterValues()也生效了。配合使用HttpServletRequestWrapper & OncePerRequestFilter 实现了对request的修改。

感谢阅读,希望对你有所帮助 :) 

来源:blog.csdn.net/qq_38020915/article/details/116431612

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/小舞很执着/article/detail/938146
推荐阅读
相关标签
  

闽ICP备14008679号