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. 代码提交时让svn忽略classpath、target、.project等

    在用eclipse操作时,经常用到svn的与资源同步这个操作,但是打开的时候会有很多生成的class文件,其实这些并不需要提交的,因为svn原则上是用来管理源代码的.每次资源同步时看到很多class文 ...

  2. Linux配置本地无密码访问

    本机配置无密码访问基本操作步骤: 1.ssh-keygen (效果同ssh-keygen -t rsa 一样,也可以ssh-keygen -t dsa) 2.ssh-copy-id -i ~/.ssh ...

  3. 【2016-10-17】【坚持学习】【Day8】【简单工厂模式】

    今天学习简单工厂模式, 结构 一个抽象产品 多个具体产品 一个工厂类,通过传入参数,new出不同的产品 代码: abstract class Product { //所有产品类的公共业务方法 publ ...

  4. Caffe源码解析2:SycedMem

    转载请注明出处,楼燚(yì)航的blog,http://www.cnblogs.com/louyihang loves baiyan/ 看到SyncedMem就知道,这是在做内存同步的操作.这类个类的 ...

  5. OpenFlow:Enabling Innovation in Campus Networks

    SDN领域,OpenFLow现在已经成为了广泛使用的南向接口协议.若想好好学习SDN,在这个领域有所进步,需要熟悉OpenFlow协议.我最近找了篇有关OpenFLow的论文,发现最早该协议是在Sig ...

  6. 一切Web的基础----HTTP

    HTTP 是基于 TCP/IP 协议的应用层协议.它不涉及数据包(packet)传输,主要规定了客户端和服务器之间的通信格式,默认使用80端口.HTTP协议基于TCP连接,该协议针对TCP连接上的数据 ...

  7. WARN util.NativeCodeLoader: Unable to load native-hadoop library for your platform... using builtin-java classes where applicable

    http://dl.bintray.com/sequenceiq/sequenceiq-bin/ http://www.secdoctor.com/html/yyjs/31101.html

  8. Path Sum II

    Path Sum II Given a binary tree and a sum, find all root-to-leaf paths where each path's sum equals ...

  9. 关于vs2012、tfs2012、windows server 2008r2一些记录

    windows server 2008r2安装在虚拟机中,装有tfs2012.sql server 2012. 物理机装有vs2012 1.用vs2012连接tfs时候,会让输入一个有效用户.输入的是 ...

  10. CSS背景图像位置属性background-position百分比详解

    百分比值同关键字很接近,但其操作方式不一样.用百分比值来居中一幅背景图像,也很简单: body { background-image;url(beijing.gif); background-repe ...