对请求或者响应进行拦截,并做额外的逻辑处理

filter能在一个请求访问目标资源之前对其进行拦截然后做某些逻辑处理,例如权限检查,也可以在一个输出响应到达客户端之前对其进行拦截并做一些额外的操作(例如,设置响应编码格式)。

二、接口定义

public abstract interface Filter{
public abstract void init(FilterConfig filterConfig) throws ServletException;
public abstract void doFilter(ServletRequest request, ServletResponse response, FilterChain chain) throws IOException,ServletException;
public abstract void destroy();
}

三、执行原理分析

当客户端发送一个请求访问某个资源时,Servlet容器会根据web.xml中的配置找出该请求所有匹配的过滤器,并保存到一个数组中,从而形成一个过滤器链,链中的每个过滤器都是一个处理节点。然后通过调用Tomcat服务器中的ApplicationFilterChain对象中的doFilter方法,不断迭代执行数组中的过滤器中的doFilter方法对请求和响应进行额外的逻辑处理(因为过滤器中的doFilter方法会重新调用FilterChain中的doFilter方法)。

Filter其实就是基于回调函数实现的责任链模式的应用。

模拟Servlet容器执行过滤器的场景:

import java.util.List;

class Scratch {
public static void main(String[] args) {
new ApplicationFilterChain().doFilter(new ServletRequest(), new ServletResponse());
} public static interface Filter {
void doFilter(ServletRequest request, ServletResponse response, FilterChain filterChain);
} /**
* 自定义过滤器
*/
public static class MyFilter1 implements Filter {
public void doFilter(ServletRequest request, ServletResponse response, FilterChain filterChain) {
System.out.println("即将执行过滤器1");
//回调方法,用于调用链中的下一个过滤器
filterChain.doFilter(request, response);
System.out.println("已经执行过滤器1");
}
} /**
* 自定义过滤器
*/
public static class MyFilter2 implements Filter {
public void doFilter(ServletRequest request, ServletResponse response, FilterChain filterChain) {
System.out.println("即将执行过滤器2");
//回调方法,用于调用链中的下一个过滤器
filterChain.doFilter(request, response);
System.out.println("已经执行过滤器2");
}
} /**
* 自定义过滤器
*/
public static class MyFilter3 implements Filter {
public void doFilter(ServletRequest request, ServletResponse response, FilterChain filterChain) {
System.out.println("即将执行过滤器3");
//回调方法,用于调用链中的下一个过滤器
filterChain.doFilter(request, response);
System.out.println("已经执行过滤器3");
}
} /**
* 模拟过滤器链接口
*/
public static interface FilterChain {
void doFilter(ServletRequest request, ServletResponse response);
} /**
* 模拟ApplicationFilterChain
*/
public static class ApplicationFilterChain implements FilterChain {
//模拟注册的过滤器,保存在一个数组中
Filter[] filters = new Filter[]{new MyFilter1(), new MyFilter2(), new MyFilter3()};
//用于保存遍历过滤器数组的下标
private int pos = 0; /**
* 开始执行过滤器逻辑
*/
public void doFilter(ServletRequest request, ServletResponse response) {
//pos小于过滤器数组中的元素说明过滤器还没执行完
if (pos < filters.length) {
//调用过滤器的doFilter方法,并将this这个当前对象传递进去
filters[pos++].doFilter(request, response, this);
}
}
} /**
* 模拟请求对象
*/
public static class ServletRequest { } /*
* 模拟响应对象
* */
public static class ServletResponse { }
}

结果如下:

四、Filter的生命周期

Filter类似于Servlet,其对象的创建和销毁由Servlet容器负责。当Servlet容器启动的时候,将创建Filter实例对象,并调用其init()方法进行一些初始化操作。当Servlet容器关闭或者重启时,将调用Filter对象的destory()方法,以释放相应的资源。init()和destory()方法在Filter对象的整个生命周期中都只仅会执行一次。当客户端有多个并发请求访问某个资源的时候,doFilter方法可能同时在多个线程环境中被执行,因此有可能出现并发安全问题,需要尽量确保doFilter方法中不会出现操作全局变量的情况。

五、配置Filter

1、xml方法配置

  <filter>
<filter-name>MyFilter</filter-name>
<filter-class>com.demo.web.MyFilter</filter-class>
</filter>
<servlet-mapping>
<servlet-name>MyFilter</servlet-name>
<url-pattern>/*</url-pattern>
</servlet-mapping>

使用xml配置注解时,默认配置在前面的Filter会先执行。

另外,filterMapping中还可以通过<dispatcher>节点配置指示Servlet容器拦截何种方式的资源。

1、REQUEST:只拦截用户直接访问的资源(默认)。

2、INCLUDE:只拦截通过RequestDispatcher的include()方法访问的资源。

3、FORWARD:只拦截通过RequestDispatcher的forward()方法访问的资源。

3、ERROR:只拦截通过声明式异常处理机制调用的资源。

2、使用注解配置

@WebFilter(filterName="MyFilter",urlPatterns={"/*"})

六、示例代码

MyFilter.java

package com.fengjr.demo01;

import javax.servlet.*;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException; public class MyFilter implements Filter { public void init(FilterConfig filterConfig) throws ServletException { } public void doFilter(ServletRequest servletRequest, ServletResponse servletResponse, FilterChain filterChain) throws IOException, ServletException {
servletRequest.setCharacterEncoding("utf-8");
servletResponse.setCharacterEncoding("utf-8");
servletResponse.setContentType("text/html;charset=utf-8");
servletResponse.getWriter().write("<br/>拦截请求,在执行业务逻辑方法之前<br/>");
servletRequest.setAttribute("str", "Filter");
filterChain.doFilter(servletRequest, servletResponse);
servletResponse.getWriter().write("<br/>拦截响应,执行业务逻辑之后追加此段文本。");
} public void destroy() { }
}

web.xml配置

    <filter>
<filter-name>MyFilter</filter-name>
<filter-class>com.fengjr.demo01.MyFilter</filter-class>
</filter>
<filter-mapping>
<filter-name>MyFilter</filter-name>
<url-pattern>/MyServlet</url-pattern>
</filter-mapping>

JavaWeb三大组件之Filter的更多相关文章

  1. JavaWeb三大组件(Servlet,Filter,Listener 自己整理,初学者可以借鉴一下)

    JavaWeb三大组件(Servlet,Filter,Listener 自己整理,初学者可以借鉴一下) Reference

  2. JavaWeb三大组件—过滤器filter

    JavaWeb三大组件 1. 都需要在web.xml中进行配置ServletListener(2个感知监听器不需要配置)Filter 2. 过滤器 它会在一组资源(jsp.servlet..css.. ...

  3. JavaWeb三大组件之一Filter知识总结

    [1] Filter简介    > Filter翻译为中文是过滤器的意思.    > Filter是JavaWeb的三大web组件之一Servlet.Filter.Listener    ...

  4. JavaWeb三大组件(Servlet、Filter、Listener)

    JavaWeb三大组件指的是:Servlet.Filter.Listener,这三个组件在JavaWeb开发中分别提供不同的功能,然而很多人可能只用过其中一个或者两个(Servlet.Filter,比 ...

  5. 学习笔记_过滤器概述(过滤器JavaWeb三大组件之一)

    过滤器Filter Filter和Lister是Servlet规范里的两个高级特性.不同于Servlet,它们不用于处理客户端请求,只用于对request.response进行修改或者对context ...

  6. JavaWeb三大组件

    一.JavaWeb三大组件 Servlet,Listener,Filter.它们在JavaWeb开发中分别提供不同的功能. JavaWeb三大组件都必须在Web.xml中配置 二.三大组件 1.Ser ...

  7. 转-JavaWeb三大组件之Listener监听器

    JavaWeb三大组件之Listener监听器一.概述1,它是一个接口,内容由我们来实现 2,它需要注册,例如注册在按钮上 3,监听器中的方法,会在特殊事件发生时被调用 二.JavaWeb中的监听器1 ...

  8. Filter过滤器-JavaWeb三大组件之一

    Servlet.Filter.Listener是JavaWeb的三大组件,给Web开发提供了很大的便利. 什么是Filter? Filter,过滤器.类似与生活中的净水器.空气净化器. JavaWeb ...

  9. Javaweb三大组件-过滤器、监听器

    1. 过滤器 [filter] 作用: 对单个获取多个servlet起到增强[advice]的作用. 用于在所有的servlet执行前,做一些预处理.例如:做编码处理, 访问量统计[servletCo ...

随机推荐

  1. C语言典型编程2

    关于C的一些小而精的编程,适合希望提升编程能力的初学者学习:关键编程也就几句,但思维可以迁移到其他编程语言.同一问题,算法多种. //任意整数的任意次方取后3位(算数取位)#include<st ...

  2. ArrayList 加强版的数组

    ArrayList 泛型类. 描述:可以自动扩容的数组. 特点:插入和删除慢,查找快. 现在来创建一个 要放String的ArrayList ArrayList list = new ArrayLis ...

  3. Java框架spring 学习笔记(九):Spring的bean管理(@Required、@Component、@Autowired、@Resource注解)

    注解:代码里面特殊的标记,使用注解可以完成相关功能 注解写法:@注解名称(属性名.属性值) @Required 用在set方法上,一旦用了这个注解,那么容器在初始化bean的时候必须要进行set,也就 ...

  4. php-fpm重启

    Ubuntu 18.04服务器 修改php init 文件后(/etc/php/7.2/fpm/php.ini)需要重启php-fpm,方法是: kill -USR2 `cat /run/php/ph ...

  5. webpack简笔(1)

    1.npm init -> 生成package.json文件 2.npm install webpack --save-dev (不建议全局安装 ,会锁定版本) [ --save-dev 开发环 ...

  6. docker 容器 详解

    docker run  ## 创建一个新容器 [root@localhost ~]# docker run --help Usage: docker run [OPTIONS] IMAGE [COMM ...

  7. Windows单机配置Zookeeper环境

    转自:http://www.jianshu.com/p/f7037105db46 首先要确保机器已经安装好java环境,并且配置好环境变量   http://apache.fayea.com/zook ...

  8. c#Dapper mysql按时间段查询和过滤

    #endregion /// <summary> /// 根据条件获取集合 /// </summary> /// <param name="id"&g ...

  9. github ignore 规范

    转自:https://www.cnblogs.com/xuld gitignore 应该包含 5 块内容: 当前项目需要忽略的文件 项目性质需要忽略的文件(比如是 nodejs 项目,有些文件就需要忽 ...

  10. Saliency Detection via Graph-Based Manifold Ranking

    Saliency Detection via Graph-Based Manifold Ranking https://www.yuque.com/lart/papers 本文不是按照之前的论文那样, ...