Servlet Filter过滤器执行顺序

 更新时间:2020年12月01日 14:23:26   作者:DayRain  
这篇文章主要介绍了Servlet Filter过滤器执行顺序的相关资料,帮助大家更好的理解为什么要用过滤器,感兴趣的朋友可以了解下

Servlet中的过滤器相当于守护后台资源的一道关卡,我们可以在过滤器中进行身份校验、权限认证、请求过滤等。

过滤器本身并不难,我们只需要知道他的定义方法、作用范围、执行顺序即可。

网上对于过滤器执行顺序的描述可能会让人产生误解。

图片来源于网络

客户端请求到达的时候,经过一次过滤器。

服务器处理完请求的时候,经过一次过滤器。

虽然经过两次过滤器,但不代表同样的代码执行了两次。

下面做了个简单的测试,看下执行结果就应该知道真正的执行流程了。

测试环境

tomcat9(servlet4.0)

jdk1.8

新版servlet可以通过注解注册servlet组件以及过滤器,无需再到web.xml下注册了。

测试过程

测试之间要先知道filterChain(过滤链)是干嘛的。

一个过滤器处理完后,会把request和response对象通过filterchain传递给下一个过滤器,如果没有下一个过滤器,则会直接开始执行业务代码,

单个过滤器

定义一个过滤器A

@WebFilter(value = "/*", filterName="A")
public class FilterA implements Filter {

  @Override
  public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain)
      throws IOException, ServletException {
    SimpleDateFormat format = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
    System.out.println(format.format(new Date()));
    System.out.println("A:拦截1");
    chain.doFilter(request, response);
    System.out.println(format.format(new Date()));
    System.out.println("A:拦截2");
  }

  @Override
  public void init(FilterConfig filterConfig) throws ServletException {

  }

  @Override
  public void destroy() {

  }

}

定义一个servlet,sleep5秒

@WebServlet("/mainUrl")
public class MainController extends HttpServlet {
  private static final long serialVersionUID = 1L;
    
  /**
   * @see HttpServlet#HttpServlet()
   */
  public MainController() {
    super();
    // TODO Auto-generated constructor stub
  }

  /**
   * @see HttpServlet#doGet(HttpServletRequest request, HttpServletResponse response)
   */
  protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
    // TODO Auto-generated method stub
    try {
      Thread.sleep(5000);
    } catch (InterruptedException e) {
      // TODO Auto-generated catch block
      e.printStackTrace();
    }
  }

  /**
   * @see HttpServlet#doPost(HttpServletRequest request, HttpServletResponse response)
   */
  protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
    // TODO Auto-generated method stub
    doGet(request, response);
  }

}

运行结果

2020-12-01 10:46:50
A:拦截1
2020-12-01 10:46:55
A:拦截2

执行顺序:

filterChain之前的代码 ——>业务处理——>filterChain之后的代码。

多个过滤器

servlet的注解在多个过滤器的情况下,是按照过滤器的名称来排序的,例如A开头的过滤器,在B开头的后面。

A过滤器

@WebFilter(value = "/*", filterName="A")
public class FilterA implements Filter {

  @Override
  public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain)
      throws IOException, ServletException {
    SimpleDateFormat format = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
    System.out.println(format.format(new Date()));
    System.out.println("A:拦截1");
    chain.doFilter(request, response);
    System.out.println(format.format(new Date()));
    System.out.println("A:拦截2");
  }

  @Override
  public void init(FilterConfig filterConfig) throws ServletException {

  }

  @Override
  public void destroy() {

  }

}

B过滤器

@WebFilter(value = "/*", filterName="B")
public class FilterB implements Filter{

  @Override
  public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain)
      throws IOException, ServletException {
    // TODO Auto-generated method stub
    SimpleDateFormat format = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
    System.out.println(format.format(new Date()));
    System.out.println("B:拦截1");
    chain.doFilter(request, response);
    System.out.println(format.format(new Date()));
    response.setContentType("normal content");
    System.out.println("B:拦截2");
    
  }

  @Override
  public void init(FilterConfig filterConfig) throws ServletException {

  }

  @Override
  public void destroy() {

  }
}

运行结果:

2020-12-01 10:53:00
A:拦截1
2020-12-01 10:53:00
B:拦截1
2020-12-01 10:53:05
B:拦截2
2020-12-01 10:53:05
A:拦截2

执行顺序:

B:拦截1和B:拦截2之间,停顿了5秒处理业务。所以先执行了  chain.doFilter前的 A、B过滤器代码,处理完业务返回的时候正好相反,先返回执行B的代码,再执行的A的代码。

总结

再来看这个图,可以略微改一下了。

 分界线是filterChain过滤链,请求进来的时候,执行filterchain之前的代码,返回response的时候,执行filterchain之后的代码。

多个过滤器之间的执行顺序,满足“先进后出” (栈结构)的原则。

其他

如果在测试过程中,发现过滤器执行了很多次,那么也可能是因为测试环境中包含了某些静态资源。

过滤器A

@WebFilter(value = "/*", filterName="A")
public class FilterA implements Filter {

  @Override
  public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain)
      throws IOException, ServletException {
    SimpleDateFormat format = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
    System.out.println(format.format(new Date()));
    System.out.println("A:拦截1");
    chain.doFilter(request, response);
    System.out.println(format.format(new Date()));
    System.out.println("A:拦截2");
  }

  @Override
  public void init(FilterConfig filterConfig) throws ServletException {

  }

  @Override
  public void destroy() {

  }

}

过滤器B

@WebFilter(value = "/*", filterName="B")
public class FilterB implements Filter{

  @Override
  public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain)
      throws IOException, ServletException {
    // TODO Auto-generated method stub
    HttpServletRequest req = (HttpServletRequest) request;
    SimpleDateFormat format = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
    System.out.println(req.getRequestURL());
    System.out.println(format.format(new Date()));
    System.out.println("B:拦截1");
    chain.doFilter(request, response);
    System.out.println(format.format(new Date()));
    response.setContentType("normal content");
    System.out.println("B:拦截2");
    
  }

  @Override
  public void init(FilterConfig filterConfig) throws ServletException {

  }

  @Override
  public void destroy() {

  }
}

主程序

@WebServlet("/mainUrl")
public class MainController extends HttpServlet {
  private static final long serialVersionUID = 1L;
    
  /**
   * @see HttpServlet#HttpServlet()
   */
  public MainController() {
    super();
    // TODO Auto-generated constructor stub
  }

  /**
   * @see HttpServlet#doGet(HttpServletRequest request, HttpServletResponse response)
   */
  protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
    // TODO Auto-generated method stub
    try {
      Thread.sleep(5000);
    } catch (InterruptedException e) {
      // TODO Auto-generated catch block
      e.printStackTrace();
    }
    request.getRequestDispatcher("/WEB-INF/pages/main.jsp").forward(request, response);
//    response.sendRedirect("/WEB-INF/pages/main.jsp");
  }

  /**
   * @see HttpServlet#doPost(HttpServletRequest request, HttpServletResponse response)
   */
  protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
    // TODO Auto-generated method stub
    doGet(request, response);
  }

}

执行结果:

2020-12-01 11:09:38
A:拦截1
http://localhost:8080/StudentManage/mainUrl
2020-12-01 11:09:38
B:拦截1
2020-12-01 11:09:43
B:拦截2
2020-12-01 11:09:43
A:拦截2
2020-12-01 11:09:44
A:拦截1
http://localhost:8080/StudentManage/css/bootstrap.css.map
2020-12-01 11:09:44
B:拦截1
2020-12-01 11:09:44
B:拦截2
2020-12-01 11:09:44
A:拦截2

转发(forward)的页面中需要请求静态资源,再次触发了过滤器。

以上就是Servlet Filter过滤器执行顺序的详细内容,更多关于Servlet Filter过滤器的资料请关注脚本之家其它相关文章!

相关文章

  • springmvc path请求映射到bean 方法的流程

    springmvc path请求映射到bean 方法的流程

    这篇文章主要介绍了springmvc path请求映射到bean 方法的流程,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2021-07-07
  • 使用IDEA创建SpringBoot项目的方法步骤

    使用IDEA创建SpringBoot项目的方法步骤

    这篇文章主要介绍了使用IDEA创建SpringBoot项目的方法步骤,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2018-05-05
  • Java实现简单文件过滤器功能

    Java实现简单文件过滤器功能

    下面小编就为大家分享一篇Java实现简单文件过滤器功能,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2018-01-01
  • Springboot读取templates文件html代码实例

    Springboot读取templates文件html代码实例

    这篇文章主要介绍了Springboot读取templates文件html代码实例,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下
    2020-04-04
  • Java8中字符串处理库strman-java的使用示例

    Java8中字符串处理库strman-java的使用示例

    除了Java本身的字符串处理方式外,我们还可以使用Apache Common Langs里的StringUtils来简化String的操作。但以上两种方式对于我们日常编程中最容易碰到的字符串处理来说,仍然显得有些不足。所以这篇文章给大家介绍Java8中字符串处理库strman-java的使用。
    2016-09-09
  • 在SpringBoot中实现线程池并行处理任务的方法详解

    在SpringBoot中实现线程池并行处理任务的方法详解

    在使用Spring Boot开发应用程序时,我们经常需要处理一些耗时的任务,例如网络请求、数据库操作或者其他需要花费一定时间的计算任务,本文将介绍如何在Spring Boot中使用线程池来实现任务的并行处理
    2023-06-06
  • IDEA中的JFormDesigner使用小结

    IDEA中的JFormDesigner使用小结

    JFormDesigner是一款用于设计和创建图形用户界面的插件,本文主要介绍了IDEA中的JFormDesigner使用小结,具有一定的参考价值,感兴趣的可以了解一下
    2024-01-01
  • java实现登录案例

    java实现登录案例

    这篇文章主要为大家详细介绍了java实现登录案例的相关代码,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2019-08-08
  • java开发ShardingSphere的路由引擎类型示例详解

    java开发ShardingSphere的路由引擎类型示例详解

    这篇文章主要为大家介绍了java开发ShardingSphere的路由引擎类型示例详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2022-08-08
  • java反射获取一个object属性值代码解析

    java反射获取一个object属性值代码解析

    这篇文章主要介绍了java反射获取一个object属性值代码解析,具有一定借鉴价值,需要的朋友可以参考下。
    2017-12-12

最新评论