1. 前言

Filter—Filter 技术是servlet2.3 新增加的功能。完成的流程:对用户请求进行预处理,接着将请求交给Servlet进行处理并生成响应,最后Filter再对服务器响应进行后处理。

Filter体现了一种职责琏模式。那么他是如何体现的呢?

2. 职责链模式

在具体的解释这个之前先看看职责链模式的定义:使多个对象都有机会处理请求 ,从而避免请求的发送者和接受者之间的耦合关系。将这个对象连成一个链,并沿着这条链传递请求,直到有一个对象处理它为止。

对于Filter而言就是,请求传给具体的web资源(比如jsp/servlet)之前要经过Filter的预处理,在web资源处理完成返回给客户端之前也要被Filter处理一遍。就好比下面的这种图

当创建了多个Filter之后,客户端传来一个Request请求,它就面对着这一个Filter链,职责琏模式就体现在这里。这个请求会在这个
Filter链上一个一个被传递下去对它进行预处理,处理完成之后就传给下一个Filter直到最后一个,然后才交给web进行相应的访问和处理。它的
uml图(并不是完成的结构图,只是体现职责琏模式的结构图)如下:

一个Filter接口定义了三个方法:init()(初始化方法);destroy()(销毁方法);doFilter()(核心的职责方法);两个具体类实现了Filter接口:ConcreteFilter1和ConcreteFilter2;

一个FilterChain接口定义了一个方法:doFilter();一个具体的实现类ConcreteFilterChain;

其中FilterChain主要的作用是完成找到下一个Filter。

3. 具体的实现

下面是对于上面结构图的一个简单实现,帮助我们理解一下Filter体现的职责琏模式。

ConcreteFilter1类

  1. package com.test.filter;
  2. import java.io.IOException;
  3. import javax.servlet.Filter;
  4. import javax.servlet.FilterChain;
  5. import javax.servlet.FilterConfig;
  6. import javax.servlet.ServletException;
  7. import javax.servlet.ServletRequest;
  8. import javax.servlet.ServletResponse;
  9. /**
  10. * ConcreteFilter1类
  11. * @author pf
  12. *
  13. */
  14. public class ConcreteFilter1 implements Filter {
  15. //  private String encoding;
  16. @Override
  17. public void destroy() {
  18. System.out.println("ConcreteFilter1()的destroy()执行");
  19. }
  20. @Override
  21. public void doFilter(ServletRequest request, ServletResponse response,
  22. FilterChain chain) throws IOException, ServletException {
  23. System.out.println("----ConcreteFilter1()的chain.doFilter()调用之前:对用户请求(request)进行预处理");
  24. //继续执行
  25. //后面有filter继续调用,没有的话就进入到了jsp,一直调用最后
  26. chain.doFilter(request, response);
  27. System.out.println("ConcreteFilter1()的chain.doFilter()调用之后:对服务器响应(response)进行后处理");
  28. }
  29. @Override
  30. public void init(FilterConfig filterConfig) throws ServletException {
  31. //      System.out.println("init开始");
  32. //      this.encoding = filterConfig.getInitParameter("encoding");
  33. //      System.out.println("init得到encoding:" + encoding);
  34. System.out.println("ConcreteFilter1()的init()方法调用");
  35. }
  36. }

ConcreteFilter2类

  1. package com.test.filter;
  2. import java.io.IOException;
  3. import javax.servlet.Filter;
  4. import javax.servlet.FilterChain;
  5. import javax.servlet.FilterConfig;
  6. import javax.servlet.ServletException;
  7. import javax.servlet.ServletRequest;
  8. import javax.servlet.ServletResponse;
  9. /**
  10. * 采用Filter统一处理字符集
  11. * @author pf
  12. *
  13. */
  14. public class ConcreteFilter2 implements Filter {
  15. //  private String encoding;
  16. @Override
  17. public void destroy() {
  18. System.out.println("ConcreteFilter2()的destroy()执行");
  19. }
  20. @Override
  21. public void doFilter(ServletRequest request, ServletResponse response,
  22. FilterChain chain) throws IOException, ServletException {
  23. System.out.println("ConcreteFilter2()的chain.doFilter()调用之前");
  24. //      request.setCharacterEncoding(encoding);
  25. //后面有filter继续调用,没有的话就进入到了jsp,一直调用最后
  26. chain.doFilter(request, response);
  27. System.out.println("ConcreteFilter2()的chain.doFilter()调用完成");
  28. }
  29. @Override
  30. public void init(FilterConfig filterConfig) throws ServletException {
  31. //      System.out.println("init开始");
  32. //      this.encoding = filterConfig.getInitParameter("encoding");
  33. //      System.out.println("init得到encoding:" + encoding);
  34. System.out.println("ConcreteFilter2()的init()方法调用");
  35. }
  36. }

web.xml

  1. <?xml version="1.0" encoding="UTF-8"?>
  2. <web-app version="2.4"
  3. xmlns="http://java.sun.com/xml/ns/j2ee"
  4. xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
  5. xsi:schemaLocation="http://java.sun.com/xml/ns/j2ee
  6. http://java.sun.com/xml/ns/j2ee/web-app_2_4.xsd">
  7. <filter>
  8. <filter-name>ConcreteFilter1</filter-name>
  9. <filter-class>com.test.filter.ConcreteFilter1</filter-class>
  10. </filter>
  11. <filter-mapping>
  12. <filter-name>ConcreteFilter1</filter-name>
  13. <url-pattern>*.jsp</url-pattern>
  14. </filter-mapping>
  15. <filter>
  16. <filter-name>ConcreteFilter2</filter-name>
  17. <filter-class>com.test.filter.ConcreteFilter2</filter-class>
  18. </filter>
  19. <filter-mapping>
  20. <filter-name>ConcreteFilter2</filter-name>
  21. <url-pattern>*.jsp</url-pattern>
  22. </filter-mapping>
  23. </web-app>

完成之后启动服务器,我这里是tomcat,会显示如下信息:

说明在启动服务器的时候就会创建我们的Filter对象

随便访问一个页面再次查看console

关闭服务器:说明在关闭服务器的时候销毁创建的Filter对象

从上面的执行结果再来看一下Filter。我们设置了两个Filter,分别是ConcreteFilter1和ConcreteFilter2.按照
我们在web.xml中配置的顺序来执行,先执行了ConcreteFilter1,在执行ConcreteFilter2.

但是注意观察,他们在真正调用chain的doFilter方法之后的调用顺序正好相反了。所以我们从这个结果可以看到Filter的执行顺序是遵循”后
进先出”的原则。现将传来的url按照配置中的顺序进行预处理,但是确实先按照相反的filter顺序执行处理好的请求。

下面是讲他的调用过程画了一个时序图:

3. 总结:

通过上面代码执行的结果来看,Filter很好的实现了职责链模式,对于任何一个请求来讲都有一条Filter链可以处理它,具体是哪一个处理了我们其实并不知道但是在到达servlet之前就是已经给我们处理好了,这样子就很好的做到了对象之间的解耦和。

原文:Filter体现职责链模式

Filter体现职责链模式的更多相关文章

  1. Filter技术+职责链模式

    Filter是一个过滤器,存在Webclient与请求的资源之间.这里的资源能够说是jsp或servlet.它的作用就是在请求达到资源之前,先对请求进行预处理.而且也能够对servlet处理后的res ...

  2. 职责链模式(chain of responsibility)

    一. 写在前面的 这么多的设计模式,我觉得职责链是我第一次看上去最简单,可是回想起来却又最复杂的一个模式. 因此,这个文章我酝酿了很久,一直也没有胆量发出来,例子也是改了又改,可是仍然觉得不够合理.所 ...

  3. 重温设计模式(三)——职责链模式(chain of responsibility)

    一. 写在前面的 这么多的设计模式,我觉得职责链是我第一次看上去最简单,可是回想起来却又最复杂的一个模式. 因此,这个文章我酝酿了很久,一直也没有胆量发出来,例子也是改了又改,可是仍然觉得不够合理.所 ...

  4. javascript设计模式学习之十三——职责链模式

    一.职责链的定义和使用场景 职责链模式的定义是,职责链模式将一系列可能会处理请求的对象连接成一条链,请求在这些对象之间一次传递,直到遇到一个可以处理它的对象.从而避免请求的发送者和接收者之间的耦合关系 ...

  5. js设计模式(12)---职责链模式

    0.前言 老实讲,看设计模式真得很痛苦,一则阅读过的代码太少:二则从来或者从没意识到使用过这些东西.所以我采用了看书(<js设计模式>)和阅读博客(大叔.alloyteam.聂微东)相结合 ...

  6. CSharp设计模式读书笔记(14):职责链模式(学习难度:★★★☆☆,使用频率:★★☆☆☆)

    职责链模式(Chain of Responsibility  Pattern):避免请求发送者与接收者耦合在一起,让多个对象都有可能接收请求,将这些对象连接成一条链,并且沿着这条链传递请求,直到有对象 ...

  7. Java设计模式之《职责链模式》及应用场景

    原创作品,可以转载,但是请标注出处地址:http://www.cnblogs.com/V1haoge/p/6530089.html 职责链模式(称责任链模式)将请求的处理对象像一条长链一般组合起来,形 ...

  8. C#设计模式之二十一职责链模式(Chain of Responsibility Pattern)【行为型】

    一.引言   今天我们开始讲"行为型"设计模式的第八个模式,该模式是[职责链模式],英文名称是:Chain of Responsibility Pattern.让我们看看现实生活中 ...

  9. LindDotNetCore~职责链模式的应用

    回到目录 职责链模式 它是一种设计模块,主要将操作流程与具体操作解耦,让每个操作都可以设置自己的操作流程,这对于工作流应用是一个不错的选择! 下面是官方标准的定义:责任链模式是一种设计模式.在责任链模 ...

随机推荐

  1. MongoDb 创建、更新以及删除文档常用命令

    mongodb由C++写就,其名字来自humongous这个单词的中间部分,从名字可见其野心所在就是海量数据的处理.关于它的一个最简洁描述为:scalable, high-performance, o ...

  2. mysql 优化(一)

    SHOW VARIABLES like 'slow_query_log'  #查看慢查询设置  1 set GLOBAL slow_query_log_file ="d:/sql/sql_l ...

  3. 阅读Real-Time O(1) Bilateral Filtering 一文的相关感受。

    研究双边滤波有很长一段时间了,最近看了一篇Real-Time O(1) Bilateral Filtering的论文,标题很吸引人,就研读了一番,经过几天的攻读,基本已理解其思想,现将这一过程做一简单 ...

  4. ACM练手

    #include<iostream> #include<string.h> using namespace std; #define N 100 class stack { c ...

  5. TCP/IP四层模型和OSI七层模型

    TCP/IP四层模型 TCP/IP是一组协议的代名词,它还包括许多协议,组成了TCP/IP协议簇.TCP/IP协议簇分为四层,IP位于协议簇的第二层(对应OSI的第三层),TCP位于协议簇的第三层(对 ...

  6. strus2验证框架

    为什么要用验证框架? 当验证规划比较复杂时,Action类的代码江边的非常繁琐,假如我们要对电话号码进行验证,是非常麻烦的. 验证框架的优点 Struts2中内置了一个验证框架,将常用的验证规则进行了 ...

  7. java Socket编程-基于UDP

    package com.wzy.UDPTest; import java.net.DatagramPacket; import java.net.DatagramSocket; import java ...

  8. void main() && int main()

    C/C++ 中从来没有定义过void main( ) .C++ 之父说过: The definition void main( ) { /* ... * / } is not and never ha ...

  9. 深入浅出SQL笔记1–数据和表

    1.数据库的概念及组成 数据库是保存表和其他相关SQL结构的容器. 数据库是由各种各样的表构成的,一个数据库里面的表总是存在相互联系的关系. 数据库内的信息组成了表,表示由行和列构成的,行是一组能够描 ...

  10. PHP的魔法方法__set() __get()

    php的魔法方法__set()与__get() Tags: PHP 我们先来看看官方的文档如何定义他们的: public void __set(string $name, mixed $value); ...