15.Filter(过滤器)

1.管理所有WEB资源:(Jsp, Servlet, 静态图片文件或静态 html 文件等)文件等进行拦截,从而实现一些特殊的功能

2.Filter接口中有一个doFilter方法,当我们编写好Filter,并配置(WEB.XML)对哪个web资源进行拦截后,

  WEB服务器每次在调用web资源的service方法之前,都会先调用一下filter的doFilter方法,因此,在该方法内编写代码可达到如下目的:

调用目标资源之前,让一段代码执行。(如对request和response进行预处理 解决编码问题)

是否调用目标资源(即是否让用户访问web资源)。

调用目标资源之后,让一段代码执行。(如事务的提交 压缩发送到服务器的数据等)

3.Filter开发步骤:

Filter开发分为二个步骤:

  1.编写java类实现Filter接口,并实现其doFilter方法。

  2.在 web.xml 文件中使用<filter>和<filter-mapping>元素对编写的filter类进行注册,并设置它所能拦截的资源。

4.Filter执行顺序 (Filter执行链)

Filter在web.xml文件中注册的顺序,决定了其在Filter执行链中的位置,首先执行注册在最前面的Filter

1 System.out.println("filterdemo1执行前!!!");
2         
3 //在执行到chain.doFilter方法,会去看,后面是否还有过滤器,如果有,先执行下一个Filter的 doFilter,后面回过头来再执行chain.doFilter
4 chain.doFilter(request, response);  //让目标资源执行,放行
5         
6 System.out.println("filterdemo1执行后!!!");

  1.filter1执行前

  2.filter2执行前

  3.servlet执行

  4.filter2执行后

  5.filter1执行后

调用了FilterChain对象的doFilter方法,则web服务器会检查FilterChain对象中是否还有filter,

如果有,则立即调用第2个filter,如果没有,则立即调用目标资源。 /*注:chain.doFilter(request,response) 执行后,立即放行,这行代码后面的语句,会等目标执行完后,再执行*/

5.Filter生命周期(类似于servlet)由WEB服务器负责 /*创建和销毁都只执行一次*/

  1.Filter的创建(WEB服务器启动)web 应用程序启动时,web 服务器将创建Filter 的实例对象,并调用其init方法,完成对象的初始化功能

  2.Filter的销毁(WEB服务器关闭) Web容器调用destroy方法销毁Filter

  3.通过init方法的参数,可获得代表当前filter配置信息的FilterConfig对象。

6.映射Filter

  <filter-mapping>元素用于设置一个 Filter 所负责拦截的资源。一个Filter拦截的资源可通过两种方式来指定:Servlet 名称和资源访问的请求路径
  <filter-name>子元素用于设置filter的注册名称。该值必须是在<filter>元素中声明过的过滤器的名字
  <url-pattern>设置 filter 所拦截的请求路径(过滤器关联的URL样式)
  <servlet-name>指定过滤器所拦截的Servlet名称。
  <dispatcher>指定过滤器所拦截的资源被 Servlet 容器 /*调用的方式*/,可以是REQUEST,INCLUDE,FORWARD和ERROR之一,默认REQUEST。

  用户可以设置多个<dispatcher> 子元素用来指定 Filter 对资源的多种调用方式进行拦截。

/*Filter高级*/

7.包装设计模式 (当某个对象的方法不适应业务需求时,通常有2种方式可以对方法进行增强)

  1.编写子类 覆盖需增强的方法
  2.使用包装设计模式对方法进行增强(优先考虑)

 1 /*
 2 1.实现与被增强对象相同的接口 
 3 2、定义一个变量记住被增强对象
 4 3、定义一个构造器,接收被增强对象
 5 4、覆盖需要增强的方法
 6 5、对于不想增强的方法,直接调用被增强对象(目标对象)的方法
 7  */
 8  
 9 class MyCharacterEncodingRequest extends HttpServletRequestWrapper{
10     
11     private HttpServletRequest request;
12     public MyCharacterEncodingRequest(HttpServletRequest request) {
13         super(request);
14         this.request = request;
15     }
16     @Override
17     public String getParameter(String name) {
18         
19         try{
20             String value= this.request.getParameter(name);
21             if(value==null){
22                 return null;
23             }
24             
25             if(!this.request.getMethod().equalsIgnoreCase("get")) {
26                 return value;
27             }
28             
29             value = new String(value.getBytes("ISO8859-1"),this.request.getCharacterEncoding());
30             return value;
31             
32         }catch (Exception e) {
33             throw new RuntimeException(e);
34         }
35     }
36 }

    1.首先看需要被增强对象继承了什么接口或父类,编写一个类也去继承这些接口或父类。
    2.在类中定义一个变量,变量类型即需增强对象的类型。
    3.在类中定义一个构造函数,接收需增强的对象。
    4.覆盖需增强的方法,编写增强的代码。
    5.不想增强的方法,直接调用被增强对象(目标对象)的方法

8.使用Decorator设计模式增强request对象(默认有这样一个实现类HttpServletRequestWrapper,让你自由发挥)

(new一个自己的Request类,继承HttpServletRequestWrapper,然后在放行时使用(chain.doFilter(自己的request,response)))

  1.使用Decorator模式包装request对象解决get和post请求方式下的中文乱码问题

  1.使用Decorator模式包装request对象实现html标签转义功能

  3.使用Decorator模式包装request对象实现敏感字符过滤功能

如果将这三个过滤器联合起来使用(都用于增强getParameter方法),那么就相当于是把request对象包装了3次,request对象的getParameter方法经过3次重写,

使得getParameter方法的功能大大增强,可以同时解决中文乱码,html标签转义,敏感字符过滤这些需求。


9.使用Decorator设计模式增强response对象(HttpServletResponseWrapper)

  1.response增强案例——压缩响应正文内容

  2.response增强案例——缓存数据到内存

原文地址:https://www.cnblogs.com/xuzekun/p/7359481.html