一个、总结

  简单的说,Filter的作用就是拦截(Tomcat的)service(Request,Response)方法。拿到Request、Response对象进行处理。然后释放控制。继续自己主动流转。其运用的还是“分层”的思想。

  至于为什么要增加这一层,为什么要对Request、Response对象进行拦截。个人感觉根本原因是为了对多个server资源(Servlet、JSP等)的请求、对应进行一些公共的设置。这里强调的是“多个”。

假设是一个资源,就没有设置Filter的必要,由于仅仅须要在这个资源须要的地方(之前)设置Request、Response就可以。也就是“设置”放到哪里都能够,由于非常多请求、响应有非常多共同的设置,所以将共同之处抽出来作为一层。就是所谓的Filter。

作用例如以下图:

传统调用:

加上Filter:

二、FilterChain

  当几个Servlet有一些同样的设置(过滤),而另几个Servlet有其他同样的设置……,这样一个Filter是不能完毕全部的过滤,所以就须要多个Filter协作来共同完毕这项任务。

于是著名的FilterChain就出来了。那么这个FilterChain中的多个Filter是怎样协作的呢?

  每一个Filter的职能不同,当一个Servlet的请求到达后,这个请求就会沿着这条Filter链,一个一个走过,走到它须要的Filter时。这个Filter就会进行对它的Request、Response进行操作。完成后自己主动向下流转。这就是所谓的职责链模式。

  每一个Filter有3个方法:

  Ø  init(FilterConfig filterConfig)

  Ø  doFilter(ServletRequest request, ServletResponse response, FilterChain chain)

  Ø  destroy()

  三个方法(以下进行具体介绍)依次运行,一个Filter链中的多个Filter的运行顺序例如以下图:

  

  即:运行第一个过滤器的chain.doFilter()之前的代码---->第二个过滤器的chain.doFilter()之前的代码---->……---->第n个过滤器的chain.doFilter()之前的代码---->所请求servlet的service()方法中的代码——>所请求servlet的doGet()或doPost()方法中的代码---->第n个过滤器的chain.doFilter()之后的代码---->……---->第二个过滤器的chain.doFilter()之后的代码---->第一个过滤器的chain.doFilter()之后的代码。

三、主要接口与方法

  首先Servlet过滤器API包括了3个接口,它们都在javax.servlet包中,各自是Filter接口、FilterChain接口和FilterConfig接口。

使用

  Ø  全部的过滤器都必须实现Filter接口。

该接口定义了init(),doFilter(),destory()三个方法。

  Ø  FilterChain接口作为了doFilter0方法的參数:  doFilter(ServletRequest request, ServletResponse response,FilterChain chain)

  Ø  FilterConfig接口作为了init()方法的參数: init(FilterConfig filterConfig)

  也就是,过滤器类将实现Filter接口,然后使用这个过滤器类中的FilterChain和FilterConfig接口。该过滤器类的—个引用将传递给FilterChain对象,以同意过滤器把控制权传递给链中的下一个资源。FilterConfig对象将由容器提供给过滤器,以同意訪问该过滤器的初始化数据。

以下对3个接口与所包括的方法进行简要介绍:

Filter接口

  1)       public void init (FilterConfig filterConfig) throws  ServletException.

  当開始使用servlet过滤器服务时。Web容器调用此方法一次。为服务准备过滤器;即通过FilterConfig对象拿到过滤器的初始化參数。即配置文件里配置的信息。然后在须要使用过滤器的时候调用doFilter()。并传递配置信息。

  2)       public voiddoFilter(ServletRequest request, ServletResponse response, FilterChain chain) throws IOException, ServletException

  每一个过滤器都接受当前的请求和响应,且FilterChain过滤器链中的过滤器(应该都是符合条件的)都会被运行。doFilter方法中,过滤器能够对请求和响应做它想做的一切,通过调用他们的方法收集数据,或者给对象加入新的行为。

  过滤器通过传送至此方法的FilterChain參数,调用chain.doFilter()将控制权传送给下一个过滤器。当这个调用返回后,过滤器能够在它的 Filter方法的最后对响应做些其他的工作。假设过滤器想要终止请求的处理或得到对响应的全然控制,则能够不调用下一个过滤器。而将其重定向至其他一些页面。

当链中的最后一个过滤器调用chain.doFilter()方法时。将执行最初请求的Servlet。

  3)       public voiddestroy()

  假设doFilter()法里的全部线程退出或已超时,容器调用此方法。

server调用destory()以指出过滤器已结束服务。用于释

放过滤器占用的资源。

FilterChain接口

  1)     public voiddoFilter(ServletRequest request,ServletResponse response)   throws
java.io.IOException,ServletException

  此方法是由Servlet容器提供给开发人员的,用于对资源请求过滤链的依次调用,通过FilterChain调用过滤链中的下一个过滤   器,假设是最后一个过滤器,则下一个就调用目标资源。

FilterConfig接口

 FilterConfig接口检索过滤器名、初始化參数以及活动的Servlet上下文。

  1)       public java.lang.String getFilterName()

  2)       public  ServletContext  getServletContex()

  3)       public java.lang.String  getlnitParameter(java.lang.String name)

  4)       public java.util.Enumeration getlnitParameterNames()

四、配置使用

  过滤器通过Web应用程序中的配置描写叙述符web.xml文件里的明。包含部分<filter>、<filter-mapping>:

<filter>。主要包含:

  1)       <filter-name>和<f'flter-class>两个必须的子元素

  2)       <icon>、<init-param>,<display-name>,<description>这4个可选的子元素。

  <filter-name>子元素定义了—个过滤器的名字,<filter-class>指定了由容器加载的实际类,<init-param>子元素为过滤器提供初始化參数。

<filter-mapping>主要由

  1)       <filter-name>,<servlet-name>和<url-pattem>子元素组成。

<servlet-name>将过滤器映射到一个或多个Servlet上,<url-pattem>将过滤器映射到—个或多个随意特征的URL的JSP页面。

<url-pattem>的4种配置方式: 

  <filter>
<filter-name>CharsetEncodingFilter</filter-name>
<filter-class>com.bjpowernode.drp.util.filter.CharsetEncodingFilter</filter-class>
<init-param>
<param-name>encoding</param-name>
<param-value>GBK</param-value>
</init-param>
</filter> <!--精确匹配,不採用不论什么修饰符 -->
<filter-mapping>
<filter-name>CharsetEncodingFilter</filter-name>
<url-pattern>/servlet/TestServlet</url-pattern>
</filter-mapping> <!--扩展匹配,由*号和扩展名组成 -->
<filter-mapping>
<filter-name>CharsetEncodingFilter</filter-name>
<url-pattern>*.jsp</url-pattern>
</filter-mapping> <!--路径前缀匹配。包括一个文件夹和一个/* -->
<filter-mapping>
<filter-name>CharsetEncodingFilter</filter-name>
<url-pattern>/servlet/*</url-pattern>
</filter-mapping> <!--全匹配。使用/* -->
<filter-mapping>
<filter-name>CharsetEncodingFilter</filter-name>
<url-pattern>/*</url-pattern>
</filter-mapping>

五、代码演示样例:


Filter类:

package com.bjpowernode.drp.util.filter;

import java.io.IOException;

import javax.servlet.Filter;
import javax.servlet.FilterChain;
import javax.servlet.FilterConfig;
import javax.servlet.ServletException;
import javax.servlet.ServletRequest;
import javax.servlet.ServletResponse; import org.omg.CORBA.Request;
/**
* 採用Filter允许处理字符集
* @author v-wangzhip
*/
public class CharsetEncodingFilter implements Filter {
private String encoding;
@Override
public void destroy() {
// TODO Auto-generated method stub }
@Override
public void doFilter(ServletRequest request, ServletResponse response,
FilterChain chain) throws IOException, ServletException {
//System.out.println("CharsetEncodingFilter---->begin");
//设置字符集
request.setCharacterEncoding(encoding);
//继续向下运行
chain.doFilter(request, response);
//System.out.println("CharsetEncodingFilter---->end");
}
@Override
public void init(FilterConfig filterConfig) throws ServletException {
this.encoding=filterConfig.getInitParameter("encoding");
System.out.println("encoding---->"+ encoding);
}
}

Web.xml配置:

  <filter>
<filter-name>CharsetEncodingFilter</filter-name>
<filter-class>com.bjpowernode.drp.util.filter.CharsetEncodingFilter</filter-class>
<init-param>
<param-name>encoding</param-name>
<param-value>GBK</param-value>
</init-param>
</filter>
<filter-mapping>
<filter-name>CharsetEncodingFilter</filter-name>
<url-pattern>*.jsp</url-pattern>
</filter-mapping>
<filter-mapping>
<filter-name>CharsetEncodingFilter</filter-name>
<url-pattern>/servlet/*</url-pattern>
</filter-mapping>

六、总结

  个人感觉,Filter是将多个请求-响应中公共的须要设置的东西提出来插入到请求-响应之间的一个中间层。

多个Filter组成一个FilterChain。每一个Filter有自己的职能。能依照一定规则来处理到达的请求和响应,体现了职责链模式。

Filter仅仅对post请求起作用,它提供了一种声明式服务。具有可插拔能力,即能够任意开启和关闭。

版权声明:本文博客原创文章,博客,未经同意,不得转载。

最新文章

  1. 设计模式-观察者模式(Observer Model)
  2. 通过git在github上建立gh-pages分支并查看网页效果
  3. 用事实说话,成熟的ORM性能不是瓶颈,灵活性不是问题:EF5.0、PDF.NET5.0、Dapper原理分析与测试手记
  4. JqueryDemoTools-用于整理jQueryDemo 分类: C# 公共资源 2014-12-02 16:50 224人阅读 评论(1) 收藏
  5. ACM竞赛高手比其他程序员水平高很多吗?
  6. 111. Minimum Depth of Binary Tree
  7. 谓词的使用 -ios
  8. 如何把python最小化安装在客户机上面
  9. SVN更新、清理乱码解决
  10. HDU 4605 Magic Ball Game 树状数组
  11. APIO2015题解
  12. SAP SOAMANAGER 配置WEBSERVICE 提示:Service cannot be reached解决方法
  13. Mysql性能优化之缓存参数优化
  14. 【POJ1753】Flip Game
  15. Thinkphp 连接数据库、查询、添加
  16. Object.defineproperty实现数据和视图的联动 ------是不是就是 Angular 模型和视图的同步的实现方式???
  17. 微信小程序保存图片到相册
  18. MySQL数据库使用规范
  19. java发送post请求,使用multipart/form-data的方式传递参数,可实现服务器间文件上传功能(转)
  20. js 常用正则表达式

热门文章

  1. 张佩的Dump服务
  2. 关于VC6编译LPDIRECT3D8不通过的问题
  3. Spring MVC集成Tiles使用方法
  4. OCA读书笔记(7) - 管理数据库存储结构
  5. android中,如果使用imageButton可以在drawable 中设置一个selector,但是imageView设置不起作用
  6. Cocos2d-x3.3RC0的Android编译Activity启动流程分析
  7. 联系人数据库设计之AbstractContactsProvider
  8. 开发自己PHP MVC框架(一)
  9. &lt;Win32_17&gt;集音频和视频播放功能于一身的简易播放器
  10. java特权制度设计篇